xref: /petsc/src/sys/fileio/ftest.c (revision bf31d2d34e67170cccc52032cb9ed4707f3cb2ab)
1e5c89e4eSSatish Balay 
2c6db04a5SJed Brown #include <petscsys.h>
37fe8d12eSJed Brown #include <errno.h>
4e5c89e4eSSatish Balay #if defined(PETSC_HAVE_PWD_H)
5e5c89e4eSSatish Balay   #include <pwd.h>
6e5c89e4eSSatish Balay #endif
7e5c89e4eSSatish Balay #include <ctype.h>
8e5c89e4eSSatish Balay #include <sys/stat.h>
9e5c89e4eSSatish Balay #if defined(PETSC_HAVE_UNISTD_H)
10e5c89e4eSSatish Balay   #include <unistd.h>
11e5c89e4eSSatish Balay #endif
12e5c89e4eSSatish Balay #if defined(PETSC_HAVE_SYS_UTSNAME_H)
13e5c89e4eSSatish Balay   #include <sys/utsname.h>
14e5c89e4eSSatish Balay #endif
15e5c89e4eSSatish Balay #if defined(PETSC_HAVE_IO_H)
16e5c89e4eSSatish Balay   #include <io.h>
17e5c89e4eSSatish Balay #endif
18e5c89e4eSSatish Balay #if defined(PETSC_HAVE_SYS_SYSTEMINFO_H)
19e5c89e4eSSatish Balay   #include <sys/systeminfo.h>
20e5c89e4eSSatish Balay #endif
21e5c89e4eSSatish Balay 
22e5c89e4eSSatish Balay #if defined(PETSC_HAVE__ACCESS) || defined(PETSC_HAVE_ACCESS)
23e5c89e4eSSatish Balay 
24*bf31d2d3SBarry Smith   #include <errno.h>
25d71ae5a4SJacob Faibussowitsch static PetscErrorCode PetscTestOwnership(const char fname[], char mode, uid_t fuid, gid_t fgid, int fmode, PetscBool *flg)
26d71ae5a4SJacob Faibussowitsch {
27e5c89e4eSSatish Balay   int m = R_OK;
28e5c89e4eSSatish Balay 
29e5c89e4eSSatish Balay   PetscFunctionBegin;
30e5c89e4eSSatish Balay   if (mode == 'r') m = R_OK;
31e5c89e4eSSatish Balay   else if (mode == 'w') m = W_OK;
32e5c89e4eSSatish Balay   else if (mode == 'x') m = X_OK;
33e32f2f54SBarry Smith   else SETERRQ(PETSC_COMM_SELF, PETSC_ERR_ARG_WRONG, "Mode must be one of r, w, or x");
34e5c89e4eSSatish Balay   #if defined(PETSC_HAVE_ACCESS)
357fe8d12eSJed Brown   if (!access(fname, m)) {
369566063dSJacob Faibussowitsch     PetscCall(PetscInfo(NULL, "System call access() succeeded on file %s\n", fname));
377fe8d12eSJed Brown     *flg = PETSC_TRUE;
387fe8d12eSJed Brown   } else {
39*bf31d2d3SBarry Smith     PetscCall(PetscInfo(NULL, "System call access() failed on file %s due to \"%s\"\n", fname, strerror(errno)));
407fe8d12eSJed Brown     *flg = PETSC_FALSE;
417fe8d12eSJed Brown   }
42e5c89e4eSSatish Balay   #else
4308401ef6SPierre Jolivet   PetscCheck(m != X_OK, PETSC_COMM_SELF, PETSC_ERR_SUP, "Unable to check execute permission for file %s", fname);
44e5c89e4eSSatish Balay   if (!_access(fname, m)) *flg = PETSC_TRUE;
45e5c89e4eSSatish Balay   #endif
463ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
47e5c89e4eSSatish Balay }
48e5c89e4eSSatish Balay 
49e5c89e4eSSatish Balay #else /* PETSC_HAVE_ACCESS or PETSC_HAVE__ACCESS */
50e5c89e4eSSatish Balay 
51d71ae5a4SJacob Faibussowitsch static PetscErrorCode PetscTestOwnership(const char fname[], char mode, uid_t fuid, gid_t fgid, int fmode, PetscBool *flg)
52d71ae5a4SJacob Faibussowitsch {
53e5c89e4eSSatish Balay   uid_t  uid;
540298fd71SBarry Smith   gid_t *gid = NULL;
55e5c89e4eSSatish Balay   int    numGroups;
56e5c89e4eSSatish Balay   int    rbit = S_IROTH;
57e5c89e4eSSatish Balay   int    wbit = S_IWOTH;
58e5c89e4eSSatish Balay   int    ebit = S_IXOTH;
592da392ccSBarry Smith   #if !defined(PETSC_MISSING_GETGROUPS)
602da392ccSBarry Smith   int    err;
612da392ccSBarry Smith   #endif
62e5c89e4eSSatish Balay 
63e5c89e4eSSatish Balay   PetscFunctionBegin;
64e5c89e4eSSatish Balay   /* Get the number of supplementary group IDs */
65e5c89e4eSSatish Balay   #if !defined(PETSC_MISSING_GETGROUPS)
669371c9d4SSatish Balay   numGroups = getgroups(0, gid);
67*bf31d2d3SBarry Smith   PetscCheck(numGroups >= 0, PETSC_COMM_SELF, PETSC_ERR_SYS, "Unable to count supplementary group IDs due to \"%s\"", strerror(errno));
689566063dSJacob Faibussowitsch   PetscCall(PetscMalloc1(numGroups + 1, &gid));
69e5c89e4eSSatish Balay   #else
70e5c89e4eSSatish Balay   numGroups = 0;
71e5c89e4eSSatish Balay   #endif
72e5c89e4eSSatish Balay 
73e5c89e4eSSatish Balay   /* Get the (effective) user and group of the caller */
74e5c89e4eSSatish Balay   uid    = geteuid();
75e5c89e4eSSatish Balay   gid[0] = getegid();
76e5c89e4eSSatish Balay 
77e5c89e4eSSatish Balay   /* Get supplementary group IDs */
78e5c89e4eSSatish Balay   #if !defined(PETSC_MISSING_GETGROUPS)
799371c9d4SSatish Balay   err = getgroups(numGroups, gid + 1);
80*bf31d2d3SBarry Smith   PetscCheck(err >= 0, PETSC_COMM_SELF, PETSC_ERR_SYS, "Unable to obtain supplementary group IDs due to \"%s\"", strerror(errno));
81e5c89e4eSSatish Balay   #endif
82e5c89e4eSSatish Balay 
83e5c89e4eSSatish Balay   /* Test for accessibility */
84e5c89e4eSSatish Balay   if (fuid == uid) {
85e5c89e4eSSatish Balay     rbit = S_IRUSR;
86e5c89e4eSSatish Balay     wbit = S_IWUSR;
87e5c89e4eSSatish Balay     ebit = S_IXUSR;
88e5c89e4eSSatish Balay   } else {
89e5c89e4eSSatish Balay     int g;
90e5c89e4eSSatish Balay 
91e5c89e4eSSatish Balay     for (g = 0; g <= numGroups; g++) {
92e5c89e4eSSatish Balay       if (fgid == gid[g]) {
93e5c89e4eSSatish Balay         rbit = S_IRGRP;
94e5c89e4eSSatish Balay         wbit = S_IWGRP;
95e5c89e4eSSatish Balay         ebit = S_IXGRP;
96e5c89e4eSSatish Balay         break;
97e5c89e4eSSatish Balay       }
98e5c89e4eSSatish Balay     }
99e5c89e4eSSatish Balay   }
1009566063dSJacob Faibussowitsch   PetscCall(PetscFree(gid));
101e5c89e4eSSatish Balay 
102e5c89e4eSSatish Balay   if (mode == 'r') {
103e5c89e4eSSatish Balay     if (fmode & rbit) *flg = PETSC_TRUE;
104e5c89e4eSSatish Balay   } else if (mode == 'w') {
105e5c89e4eSSatish Balay     if (fmode & wbit) *flg = PETSC_TRUE;
106e5c89e4eSSatish Balay   } else if (mode == 'x') {
107e5c89e4eSSatish Balay     if (fmode & ebit) *flg = PETSC_TRUE;
108e5c89e4eSSatish Balay   }
1093ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
110e5c89e4eSSatish Balay }
111e5c89e4eSSatish Balay 
112e5c89e4eSSatish Balay #endif /* PETSC_HAVE_ACCESS */
113e5c89e4eSSatish Balay 
114d71ae5a4SJacob Faibussowitsch static PetscErrorCode PetscGetFileStat(const char fname[], uid_t *fileUid, gid_t *fileGid, int *fileMode, PetscBool *exists)
115d71ae5a4SJacob Faibussowitsch {
116e5c89e4eSSatish Balay   struct stat statbuf;
1173ba16761SJacob Faibussowitsch   int         ierr;
118e5c89e4eSSatish Balay 
119e5c89e4eSSatish Balay   PetscFunctionBegin;
12038ea73c8SJed Brown   *fileMode = 0;
12138ea73c8SJed Brown   *exists   = PETSC_FALSE;
122e5c89e4eSSatish Balay #if defined(PETSC_HAVE_STAT_NO_CONST)
123e5c89e4eSSatish Balay   ierr = stat((char *)fname, &statbuf);
124e5c89e4eSSatish Balay #else
125e5c89e4eSSatish Balay   ierr = stat(fname, &statbuf);
126e5c89e4eSSatish Balay #endif
127e5c89e4eSSatish Balay   if (ierr) {
1287ad82f04SSatish Balay #if defined(EOVERFLOW)
12908401ef6SPierre Jolivet     PetscCheck(errno != EOVERFLOW, PETSC_COMM_SELF, PETSC_ERR_SYS, "EOVERFLOW in stat(), configure PETSc --with-large-file-io=1 to support files larger than 2GiB");
1307ad82f04SSatish Balay #endif
131*bf31d2d3SBarry Smith     PetscCall(PetscInfo(NULL, "System call stat() failed on file %s due to \"%s\"\n", fname, strerror(errno)));
132e5c89e4eSSatish Balay     *exists = PETSC_FALSE;
133e5c89e4eSSatish Balay   } else {
1349566063dSJacob Faibussowitsch     PetscCall(PetscInfo(NULL, "System call stat() succeeded on file %s\n", fname));
135e5c89e4eSSatish Balay     *exists   = PETSC_TRUE;
136e5c89e4eSSatish Balay     *fileUid  = statbuf.st_uid;
137e5c89e4eSSatish Balay     *fileGid  = statbuf.st_gid;
138e5c89e4eSSatish Balay     *fileMode = statbuf.st_mode;
139e5c89e4eSSatish Balay   }
1403ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
141e5c89e4eSSatish Balay }
142e5c89e4eSSatish Balay 
143e1d001d6SBarry Smith /*@C
14426b19047SBarry Smith    PetscTestFile - checks for the existence of a file
14526b19047SBarry Smith 
14626b19047SBarry Smith    Not Collective
14726b19047SBarry Smith 
148d8d19677SJose E. Roman    Input Parameters:
14926b19047SBarry Smith +  fname - the filename
15055819941SStefano Zampini -  mode - either 'r', 'w', 'x' or '\0'
15126b19047SBarry Smith 
15226b19047SBarry Smith    Output Parameter:
15326b19047SBarry Smith .  flg - the file exists and satisfies the mode
15426b19047SBarry Smith 
1551b266c99SBarry Smith    Level: intermediate
1561b266c99SBarry Smith 
157811af0c4SBarry Smith    Note:
158811af0c4SBarry Smith    If mode is '\0', no permissions checks are performed
15955819941SStefano Zampini 
160db781477SPatrick Sanan .seealso: `PetscTestDirectory()`, `PetscLs()`
16126b19047SBarry Smith @*/
162d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscTestFile(const char fname[], char mode, PetscBool *flg)
163d71ae5a4SJacob Faibussowitsch {
164e5c89e4eSSatish Balay   uid_t     fuid;
165e5c89e4eSSatish Balay   gid_t     fgid;
166e5c89e4eSSatish Balay   int       fmode;
167ace3abfcSBarry Smith   PetscBool exists;
168e5c89e4eSSatish Balay 
169e5c89e4eSSatish Balay   PetscFunctionBegin;
170e5c89e4eSSatish Balay   *flg = PETSC_FALSE;
1713ba16761SJacob Faibussowitsch   if (!fname) PetscFunctionReturn(PETSC_SUCCESS);
172e5c89e4eSSatish Balay 
1739566063dSJacob Faibussowitsch   PetscCall(PetscGetFileStat(fname, &fuid, &fgid, &fmode, &exists));
1743ba16761SJacob Faibussowitsch   if (!exists) PetscFunctionReturn(PETSC_SUCCESS);
1757fe8d12eSJed Brown   /* Except for systems that have this broken stat macros (rare), this is the correct way to check for a regular file */
1763ba16761SJacob Faibussowitsch   if (!S_ISREG(fmode)) PetscFunctionReturn(PETSC_SUCCESS);
17755819941SStefano Zampini   /* return if asked to check for existence only */
1789371c9d4SSatish Balay   if (mode == '\0') {
1799371c9d4SSatish Balay     *flg = exists;
1803ba16761SJacob Faibussowitsch     PetscFunctionReturn(PETSC_SUCCESS);
1819371c9d4SSatish Balay   }
1829566063dSJacob Faibussowitsch   PetscCall(PetscTestOwnership(fname, mode, fuid, fgid, fmode, flg));
1833ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
184e5c89e4eSSatish Balay }
185e5c89e4eSSatish Balay 
186e1d001d6SBarry Smith /*@C
18726b19047SBarry Smith    PetscTestDirectory - checks for the existence of a directory
18826b19047SBarry Smith 
18926b19047SBarry Smith    Not Collective
19026b19047SBarry Smith 
191d8d19677SJose E. Roman    Input Parameters:
19226b19047SBarry Smith +  dirname - the directory name
19326b19047SBarry Smith -  mode - either 'r', 'w', or 'x'
19426b19047SBarry Smith 
19526b19047SBarry Smith    Output Parameter:
19626b19047SBarry Smith .  flg - the directory exists and satisfies the mode
19726b19047SBarry Smith 
1981b266c99SBarry Smith    Level: intermediate
1991b266c99SBarry Smith 
200db781477SPatrick Sanan .seealso: `PetscTestFile()`, `PetscLs()`
20126b19047SBarry Smith @*/
202d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscTestDirectory(const char dirname[], char mode, PetscBool *flg)
203d71ae5a4SJacob Faibussowitsch {
204e5c89e4eSSatish Balay   uid_t     fuid;
205e5c89e4eSSatish Balay   gid_t     fgid;
206e5c89e4eSSatish Balay   int       fmode;
207ace3abfcSBarry Smith   PetscBool exists;
208e5c89e4eSSatish Balay 
209e5c89e4eSSatish Balay   PetscFunctionBegin;
210e5c89e4eSSatish Balay   *flg = PETSC_FALSE;
2113ba16761SJacob Faibussowitsch   if (!dirname) PetscFunctionReturn(PETSC_SUCCESS);
212e5c89e4eSSatish Balay 
2139566063dSJacob Faibussowitsch   PetscCall(PetscGetFileStat(dirname, &fuid, &fgid, &fmode, &exists));
2143ba16761SJacob Faibussowitsch   if (!exists) PetscFunctionReturn(PETSC_SUCCESS);
215e5c89e4eSSatish Balay   /* Except for systems that have this broken stat macros (rare), this
216e5c89e4eSSatish Balay      is the correct way to check for a directory */
2173ba16761SJacob Faibussowitsch   if (!S_ISDIR(fmode)) PetscFunctionReturn(PETSC_SUCCESS);
218e5c89e4eSSatish Balay 
2199566063dSJacob Faibussowitsch   PetscCall(PetscTestOwnership(dirname, mode, fuid, fgid, fmode, flg));
2203ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
221e5c89e4eSSatish Balay }
222e5c89e4eSSatish Balay 
223e1d001d6SBarry Smith /*@C
22426b19047SBarry Smith    PetscLs - produce a listing of the files in a directory
22526b19047SBarry Smith 
226d083f849SBarry Smith    Collective
22726b19047SBarry Smith 
228d8d19677SJose E. Roman    Input Parameters:
22926b19047SBarry Smith +  comm - the MPI communicator
23026b19047SBarry Smith .  dirname - the directory name
23126b19047SBarry Smith -  tlen - the length of the buffer found[]
23226b19047SBarry Smith 
233d8d19677SJose E. Roman    Output Parameters:
23426b19047SBarry Smith +  found - listing of files
23526b19047SBarry Smith -  flg - the directory exists
23626b19047SBarry Smith 
2371b266c99SBarry Smith    Level: intermediate
2381b266c99SBarry Smith 
239db781477SPatrick Sanan .seealso: `PetscTestFile()`, `PetscLs()`
24026b19047SBarry Smith @*/
241d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscLs(MPI_Comm comm, const char dirname[], char found[], size_t tlen, PetscBool *flg)
242d71ae5a4SJacob Faibussowitsch {
243e5c89e4eSSatish Balay   size_t len;
244e5c89e4eSSatish Balay   char  *f, program[PETSC_MAX_PATH_LEN];
245e5c89e4eSSatish Balay   FILE  *fp;
246e5c89e4eSSatish Balay 
247e5c89e4eSSatish Balay   PetscFunctionBegin;
248c6a7a370SJeremy L Thompson   PetscCall(PetscStrncpy(program, "ls ", sizeof(program)));
249c6a7a370SJeremy L Thompson   PetscCall(PetscStrlcat(program, dirname, sizeof(program)));
250e5c89e4eSSatish Balay #if defined(PETSC_HAVE_POPEN)
2519566063dSJacob Faibussowitsch   PetscCall(PetscPOpen(comm, NULL, program, "r", &fp));
252e5c89e4eSSatish Balay #else
253e32f2f54SBarry Smith   SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP_SYS, "Cannot run external programs on this machine");
254e5c89e4eSSatish Balay #endif
255e5c89e4eSSatish Balay   f = fgets(found, tlen, fp);
256a297a907SKarl Rupp   if (f) *flg = PETSC_TRUE;
257a297a907SKarl Rupp   else *flg = PETSC_FALSE;
258e5c89e4eSSatish Balay   while (f) {
2599566063dSJacob Faibussowitsch     PetscCall(PetscStrlen(found, &len));
260e5c89e4eSSatish Balay     f = fgets(found + len, tlen - len, fp);
261e5c89e4eSSatish Balay   }
2629566063dSJacob Faibussowitsch   if (*flg) PetscCall(PetscInfo(NULL, "ls on %s gives \n%s\n", dirname, found));
263e5c89e4eSSatish Balay #if defined(PETSC_HAVE_POPEN)
2649566063dSJacob Faibussowitsch   PetscCall(PetscPClose(comm, fp));
265e5c89e4eSSatish Balay #else
266e32f2f54SBarry Smith   SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP_SYS, "Cannot run external programs on this machine");
267e5c89e4eSSatish Balay #endif
2683ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
269e5c89e4eSSatish Balay }
270