1c6db04a5SJed Brown #include <petscsys.h>
27fe8d12eSJed Brown #include <errno.h>
3e5c89e4eSSatish Balay #if defined(PETSC_HAVE_PWD_H)
4e5c89e4eSSatish Balay #include <pwd.h>
5e5c89e4eSSatish Balay #endif
6e5c89e4eSSatish Balay #include <ctype.h>
7e5c89e4eSSatish Balay #include <sys/stat.h>
8e5c89e4eSSatish Balay #if defined(PETSC_HAVE_UNISTD_H)
9e5c89e4eSSatish Balay #include <unistd.h>
10e5c89e4eSSatish Balay #endif
11e5c89e4eSSatish Balay #if defined(PETSC_HAVE_SYS_UTSNAME_H)
12e5c89e4eSSatish Balay #include <sys/utsname.h>
13e5c89e4eSSatish Balay #endif
14e5c89e4eSSatish Balay #if defined(PETSC_HAVE_IO_H)
15e5c89e4eSSatish Balay #include <io.h>
16e5c89e4eSSatish Balay #endif
17e5c89e4eSSatish Balay #if defined(PETSC_HAVE_SYS_SYSTEMINFO_H)
18e5c89e4eSSatish Balay #include <sys/systeminfo.h>
19e5c89e4eSSatish Balay #endif
20e5c89e4eSSatish Balay
21e5c89e4eSSatish Balay #if defined(PETSC_HAVE__ACCESS) || defined(PETSC_HAVE_ACCESS)
22e5c89e4eSSatish Balay
23bf31d2d3SBarry Smith #include <errno.h>
PetscTestOwnership(const char fname[],char mode,uid_t fuid,gid_t fgid,int fmode,PetscBool * flg)24d71ae5a4SJacob Faibussowitsch static PetscErrorCode PetscTestOwnership(const char fname[], char mode, uid_t fuid, gid_t fgid, int fmode, PetscBool *flg)
25d71ae5a4SJacob Faibussowitsch {
26e5c89e4eSSatish Balay int m = R_OK;
27e5c89e4eSSatish Balay
28e5c89e4eSSatish Balay PetscFunctionBegin;
29e5c89e4eSSatish Balay if (mode == 'r') m = R_OK;
30e5c89e4eSSatish Balay else if (mode == 'w') m = W_OK;
31e5c89e4eSSatish Balay else if (mode == 'x') m = X_OK;
32e32f2f54SBarry Smith else SETERRQ(PETSC_COMM_SELF, PETSC_ERR_ARG_WRONG, "Mode must be one of r, w, or x");
33e5c89e4eSSatish Balay #if defined(PETSC_HAVE_ACCESS)
347fe8d12eSJed Brown if (!access(fname, m)) {
359566063dSJacob Faibussowitsch PetscCall(PetscInfo(NULL, "System call access() succeeded on file %s\n", fname));
367fe8d12eSJed Brown *flg = PETSC_TRUE;
377fe8d12eSJed Brown } else {
38bf31d2d3SBarry Smith PetscCall(PetscInfo(NULL, "System call access() failed on file %s due to \"%s\"\n", fname, strerror(errno)));
397fe8d12eSJed Brown *flg = PETSC_FALSE;
407fe8d12eSJed Brown }
41e5c89e4eSSatish Balay #else
4208401ef6SPierre Jolivet PetscCheck(m != X_OK, PETSC_COMM_SELF, PETSC_ERR_SUP, "Unable to check execute permission for file %s", fname);
43e5c89e4eSSatish Balay if (!_access(fname, m)) *flg = PETSC_TRUE;
44e5c89e4eSSatish Balay #endif
453ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
46e5c89e4eSSatish Balay }
47e5c89e4eSSatish Balay
48e5c89e4eSSatish Balay #else /* PETSC_HAVE_ACCESS or PETSC_HAVE__ACCESS */
49e5c89e4eSSatish Balay
PetscTestOwnership(const char fname[],char mode,uid_t fuid,gid_t fgid,int fmode,PetscBool * flg)50d71ae5a4SJacob Faibussowitsch static PetscErrorCode PetscTestOwnership(const char fname[], char mode, uid_t fuid, gid_t fgid, int fmode, PetscBool *flg)
51d71ae5a4SJacob Faibussowitsch {
52e5c89e4eSSatish Balay uid_t uid;
530298fd71SBarry Smith gid_t *gid = NULL;
54e5c89e4eSSatish Balay int numGroups;
55e5c89e4eSSatish Balay int rbit = S_IROTH;
56e5c89e4eSSatish Balay int wbit = S_IWOTH;
57e5c89e4eSSatish Balay int ebit = S_IXOTH;
582da392ccSBarry Smith #if !defined(PETSC_MISSING_GETGROUPS)
592da392ccSBarry Smith int err;
602da392ccSBarry Smith #endif
61e5c89e4eSSatish Balay
62e5c89e4eSSatish Balay PetscFunctionBegin;
63e5c89e4eSSatish Balay /* Get the number of supplementary group IDs */
64e5c89e4eSSatish Balay #if !defined(PETSC_MISSING_GETGROUPS)
659371c9d4SSatish Balay numGroups = getgroups(0, gid);
66bf31d2d3SBarry Smith PetscCheck(numGroups >= 0, PETSC_COMM_SELF, PETSC_ERR_SYS, "Unable to count supplementary group IDs due to \"%s\"", strerror(errno));
679566063dSJacob Faibussowitsch PetscCall(PetscMalloc1(numGroups + 1, &gid));
68e5c89e4eSSatish Balay #else
69e5c89e4eSSatish Balay numGroups = 0;
70e5c89e4eSSatish Balay #endif
71e5c89e4eSSatish Balay
72e5c89e4eSSatish Balay /* Get the (effective) user and group of the caller */
73e5c89e4eSSatish Balay uid = geteuid();
74e5c89e4eSSatish Balay gid[0] = getegid();
75e5c89e4eSSatish Balay
76e5c89e4eSSatish Balay /* Get supplementary group IDs */
77e5c89e4eSSatish Balay #if !defined(PETSC_MISSING_GETGROUPS)
789371c9d4SSatish Balay err = getgroups(numGroups, gid + 1);
79bf31d2d3SBarry Smith PetscCheck(err >= 0, PETSC_COMM_SELF, PETSC_ERR_SYS, "Unable to obtain supplementary group IDs due to \"%s\"", strerror(errno));
80e5c89e4eSSatish Balay #endif
81e5c89e4eSSatish Balay
82e5c89e4eSSatish Balay /* Test for accessibility */
83e5c89e4eSSatish Balay if (fuid == uid) {
84e5c89e4eSSatish Balay rbit = S_IRUSR;
85e5c89e4eSSatish Balay wbit = S_IWUSR;
86e5c89e4eSSatish Balay ebit = S_IXUSR;
87e5c89e4eSSatish Balay } else {
88e5c89e4eSSatish Balay int g;
89e5c89e4eSSatish Balay
90e5c89e4eSSatish Balay for (g = 0; g <= numGroups; g++) {
91e5c89e4eSSatish Balay if (fgid == gid[g]) {
92e5c89e4eSSatish Balay rbit = S_IRGRP;
93e5c89e4eSSatish Balay wbit = S_IWGRP;
94e5c89e4eSSatish Balay ebit = S_IXGRP;
95e5c89e4eSSatish Balay break;
96e5c89e4eSSatish Balay }
97e5c89e4eSSatish Balay }
98e5c89e4eSSatish Balay }
999566063dSJacob Faibussowitsch PetscCall(PetscFree(gid));
100e5c89e4eSSatish Balay
101e5c89e4eSSatish Balay if (mode == 'r') {
102e5c89e4eSSatish Balay if (fmode & rbit) *flg = PETSC_TRUE;
103e5c89e4eSSatish Balay } else if (mode == 'w') {
104e5c89e4eSSatish Balay if (fmode & wbit) *flg = PETSC_TRUE;
105e5c89e4eSSatish Balay } else if (mode == 'x') {
106e5c89e4eSSatish Balay if (fmode & ebit) *flg = PETSC_TRUE;
107e5c89e4eSSatish Balay }
1083ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
109e5c89e4eSSatish Balay }
110e5c89e4eSSatish Balay
111e5c89e4eSSatish Balay #endif /* PETSC_HAVE_ACCESS */
112e5c89e4eSSatish Balay
PetscGetFileStat(const char fname[],uid_t * fileUid,gid_t * fileGid,int * fileMode,PetscBool * exists)113d71ae5a4SJacob Faibussowitsch static PetscErrorCode PetscGetFileStat(const char fname[], uid_t *fileUid, gid_t *fileGid, int *fileMode, PetscBool *exists)
114d71ae5a4SJacob Faibussowitsch {
115e5c89e4eSSatish Balay struct stat statbuf;
1163ba16761SJacob Faibussowitsch int ierr;
117e5c89e4eSSatish Balay
118e5c89e4eSSatish Balay PetscFunctionBegin;
11938ea73c8SJed Brown *fileMode = 0;
12038ea73c8SJed Brown *exists = PETSC_FALSE;
121e5c89e4eSSatish Balay #if defined(PETSC_HAVE_STAT_NO_CONST)
122e5c89e4eSSatish Balay ierr = stat((char *)fname, &statbuf);
123e5c89e4eSSatish Balay #else
124e5c89e4eSSatish Balay ierr = stat(fname, &statbuf);
125e5c89e4eSSatish Balay #endif
126e5c89e4eSSatish Balay if (ierr) {
1277ad82f04SSatish Balay #if defined(EOVERFLOW)
12808401ef6SPierre 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");
1297ad82f04SSatish Balay #endif
130bf31d2d3SBarry Smith PetscCall(PetscInfo(NULL, "System call stat() failed on file %s due to \"%s\"\n", fname, strerror(errno)));
131e5c89e4eSSatish Balay *exists = PETSC_FALSE;
132e5c89e4eSSatish Balay } else {
1339566063dSJacob Faibussowitsch PetscCall(PetscInfo(NULL, "System call stat() succeeded on file %s\n", fname));
134e5c89e4eSSatish Balay *exists = PETSC_TRUE;
135e5c89e4eSSatish Balay *fileUid = statbuf.st_uid;
136e5c89e4eSSatish Balay *fileGid = statbuf.st_gid;
137e5c89e4eSSatish Balay *fileMode = statbuf.st_mode;
138e5c89e4eSSatish Balay }
1393ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
140e5c89e4eSSatish Balay }
141e5c89e4eSSatish Balay
1425d83a8b1SBarry Smith /*@
14326b19047SBarry Smith PetscTestFile - checks for the existence of a file
14426b19047SBarry Smith
14526b19047SBarry Smith Not Collective
14626b19047SBarry Smith
147d8d19677SJose E. Roman Input Parameters:
14826b19047SBarry Smith + fname - the filename
14955819941SStefano Zampini - mode - either 'r', 'w', 'x' or '\0'
15026b19047SBarry Smith
15126b19047SBarry Smith Output Parameter:
15226b19047SBarry Smith . flg - the file exists and satisfies the mode
15326b19047SBarry Smith
1541b266c99SBarry Smith Level: intermediate
1551b266c99SBarry Smith
156811af0c4SBarry Smith Note:
157811af0c4SBarry Smith If mode is '\0', no permissions checks are performed
15855819941SStefano Zampini
159db781477SPatrick Sanan .seealso: `PetscTestDirectory()`, `PetscLs()`
16026b19047SBarry Smith @*/
PetscTestFile(const char fname[],char mode,PetscBool * flg)161d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscTestFile(const char fname[], char mode, PetscBool *flg)
162d71ae5a4SJacob Faibussowitsch {
163e5c89e4eSSatish Balay uid_t fuid;
164e5c89e4eSSatish Balay gid_t fgid;
165e5c89e4eSSatish Balay int fmode;
166ace3abfcSBarry Smith PetscBool exists;
167e5c89e4eSSatish Balay
168e5c89e4eSSatish Balay PetscFunctionBegin;
169e5c89e4eSSatish Balay *flg = PETSC_FALSE;
1703ba16761SJacob Faibussowitsch if (!fname) PetscFunctionReturn(PETSC_SUCCESS);
171e5c89e4eSSatish Balay
1729566063dSJacob Faibussowitsch PetscCall(PetscGetFileStat(fname, &fuid, &fgid, &fmode, &exists));
1733ba16761SJacob Faibussowitsch if (!exists) PetscFunctionReturn(PETSC_SUCCESS);
1747fe8d12eSJed Brown /* Except for systems that have this broken stat macros (rare), this is the correct way to check for a regular file */
1753ba16761SJacob Faibussowitsch if (!S_ISREG(fmode)) PetscFunctionReturn(PETSC_SUCCESS);
17655819941SStefano Zampini /* return if asked to check for existence only */
1779371c9d4SSatish Balay if (mode == '\0') {
1789371c9d4SSatish Balay *flg = exists;
1793ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
1809371c9d4SSatish Balay }
1819566063dSJacob Faibussowitsch PetscCall(PetscTestOwnership(fname, mode, fuid, fgid, fmode, flg));
1823ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
183e5c89e4eSSatish Balay }
184e5c89e4eSSatish Balay
1855d83a8b1SBarry Smith /*@
18626b19047SBarry Smith PetscTestDirectory - checks for the existence of a directory
18726b19047SBarry Smith
18826b19047SBarry Smith Not Collective
18926b19047SBarry Smith
190d8d19677SJose E. Roman Input Parameters:
19126b19047SBarry Smith + dirname - the directory name
19226b19047SBarry Smith - mode - either 'r', 'w', or 'x'
19326b19047SBarry Smith
19426b19047SBarry Smith Output Parameter:
19526b19047SBarry Smith . flg - the directory exists and satisfies the mode
19626b19047SBarry Smith
1971b266c99SBarry Smith Level: intermediate
1981b266c99SBarry Smith
199bfbbc7b7SBarry Smith .seealso: `PetscTestFile()`, `PetscLs()`, `PetscRMTree()`
20026b19047SBarry Smith @*/
PetscTestDirectory(const char dirname[],char mode,PetscBool * flg)201d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscTestDirectory(const char dirname[], char mode, PetscBool *flg)
202d71ae5a4SJacob Faibussowitsch {
203e5c89e4eSSatish Balay uid_t fuid;
204e5c89e4eSSatish Balay gid_t fgid;
205e5c89e4eSSatish Balay int fmode;
206ace3abfcSBarry Smith PetscBool exists;
207e5c89e4eSSatish Balay
208e5c89e4eSSatish Balay PetscFunctionBegin;
209e5c89e4eSSatish Balay *flg = PETSC_FALSE;
2103ba16761SJacob Faibussowitsch if (!dirname) PetscFunctionReturn(PETSC_SUCCESS);
211e5c89e4eSSatish Balay
2129566063dSJacob Faibussowitsch PetscCall(PetscGetFileStat(dirname, &fuid, &fgid, &fmode, &exists));
2133ba16761SJacob Faibussowitsch if (!exists) PetscFunctionReturn(PETSC_SUCCESS);
214e5c89e4eSSatish Balay /* Except for systems that have this broken stat macros (rare), this
215e5c89e4eSSatish Balay is the correct way to check for a directory */
2163ba16761SJacob Faibussowitsch if (!S_ISDIR(fmode)) PetscFunctionReturn(PETSC_SUCCESS);
217e5c89e4eSSatish Balay
2189566063dSJacob Faibussowitsch PetscCall(PetscTestOwnership(dirname, mode, fuid, fgid, fmode, flg));
2193ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
220e5c89e4eSSatish Balay }
221e5c89e4eSSatish Balay
222e1d001d6SBarry Smith /*@C
22326b19047SBarry Smith PetscLs - produce a listing of the files in a directory
22426b19047SBarry Smith
225d083f849SBarry Smith Collective
22626b19047SBarry Smith
227d8d19677SJose E. Roman Input Parameters:
22826b19047SBarry Smith + comm - the MPI communicator
22926b19047SBarry Smith . dirname - the directory name
230bfbbc7b7SBarry Smith - tlen - the length of the buffer `found`
23126b19047SBarry Smith
232d8d19677SJose E. Roman Output Parameters:
23326b19047SBarry Smith + found - listing of files
23426b19047SBarry Smith - flg - the directory exists
23526b19047SBarry Smith
2361b266c99SBarry Smith Level: intermediate
2371b266c99SBarry Smith
238bfbbc7b7SBarry Smith .seealso: `PetscTestFile()`, `PetscRMTree()`, `PetscTestDirectory()`
23926b19047SBarry Smith @*/
PetscLs(MPI_Comm comm,const char dirname[],char found[],size_t tlen,PetscBool * flg)240d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscLs(MPI_Comm comm, const char dirname[], char found[], size_t tlen, PetscBool *flg)
241d71ae5a4SJacob Faibussowitsch {
242e5c89e4eSSatish Balay size_t len;
243e5c89e4eSSatish Balay char *f, program[PETSC_MAX_PATH_LEN];
244e5c89e4eSSatish Balay FILE *fp;
245e5c89e4eSSatish Balay
246e5c89e4eSSatish Balay PetscFunctionBegin;
247c6a7a370SJeremy L Thompson PetscCall(PetscStrncpy(program, "ls ", sizeof(program)));
248c6a7a370SJeremy L Thompson PetscCall(PetscStrlcat(program, dirname, sizeof(program)));
249e5c89e4eSSatish Balay #if defined(PETSC_HAVE_POPEN)
2509566063dSJacob Faibussowitsch PetscCall(PetscPOpen(comm, NULL, program, "r", &fp));
251e5c89e4eSSatish Balay #else
252e32f2f54SBarry Smith SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP_SYS, "Cannot run external programs on this machine");
253e5c89e4eSSatish Balay #endif
254*6497c311SBarry Smith f = fgets(found, (int)tlen, fp);
255a297a907SKarl Rupp if (f) *flg = PETSC_TRUE;
256a297a907SKarl Rupp else *flg = PETSC_FALSE;
257e5c89e4eSSatish Balay while (f) {
2589566063dSJacob Faibussowitsch PetscCall(PetscStrlen(found, &len));
259*6497c311SBarry Smith f = fgets(found + len, (int)(tlen - len), fp);
260e5c89e4eSSatish Balay }
2619566063dSJacob Faibussowitsch if (*flg) PetscCall(PetscInfo(NULL, "ls on %s gives \n%s\n", dirname, found));
262e5c89e4eSSatish Balay #if defined(PETSC_HAVE_POPEN)
2639566063dSJacob Faibussowitsch PetscCall(PetscPClose(comm, fp));
264e5c89e4eSSatish Balay #else
265e32f2f54SBarry Smith SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP_SYS, "Cannot run external programs on this machine");
266e5c89e4eSSatish Balay #endif
2673ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
268e5c89e4eSSatish Balay }
269