xref: /petsc/src/sys/fileio/ftest.c (revision d8d19677bbccf95218448bee62e6b87f4513e133)
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 
24ace3abfcSBarry Smith static PetscErrorCode PetscTestOwnership(const char fname[], char mode, uid_t fuid, gid_t fgid, int fmode, PetscBool  *flg)
25e5c89e4eSSatish Balay {
26e5c89e4eSSatish Balay   int            m = R_OK;
277fe8d12eSJed Brown   PetscErrorCode ierr;
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)) {
360298fd71SBarry Smith     ierr = PetscInfo1(NULL,"System call access() succeeded on file %s\n",fname);CHKERRQ(ierr);
377fe8d12eSJed Brown     *flg = PETSC_TRUE;
387fe8d12eSJed Brown   } else {
390298fd71SBarry Smith     ierr = PetscInfo1(NULL,"System call access() failed on file %s\n",fname);CHKERRQ(ierr);
407fe8d12eSJed Brown     *flg = PETSC_FALSE;
417fe8d12eSJed Brown   }
42e5c89e4eSSatish Balay #else
43e32f2f54SBarry Smith   if (m == X_OK) SETERRQ1(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
46e5c89e4eSSatish Balay   PetscFunctionReturn(0);
47e5c89e4eSSatish Balay }
48e5c89e4eSSatish Balay 
49e5c89e4eSSatish Balay #else  /* PETSC_HAVE_ACCESS or PETSC_HAVE__ACCESS */
50e5c89e4eSSatish Balay 
51ace3abfcSBarry Smith static PetscErrorCode PetscTestOwnership(const char fname[], char mode, uid_t fuid, gid_t fgid, int fmode, PetscBool  *flg)
52e5c89e4eSSatish Balay {
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;
59e5c89e4eSSatish Balay   PetscErrorCode ierr;
602da392ccSBarry Smith #if !defined(PETSC_MISSING_GETGROUPS)
612da392ccSBarry Smith   int            err;
622da392ccSBarry Smith #endif
63e5c89e4eSSatish Balay 
64e5c89e4eSSatish Balay   PetscFunctionBegin;
65e5c89e4eSSatish Balay   /* Get the number of supplementary group IDs */
66e5c89e4eSSatish Balay #if !defined(PETSC_MISSING_GETGROUPS)
67d8185827SBarry Smith   numGroups = getgroups(0, gid); if (numGroups < 0) SETERRQ(PETSC_COMM_SELF,PETSC_ERR_SYS, "Unable to count supplementary group IDs");
68854ce69bSBarry Smith   ierr = PetscMalloc1(numGroups+1, &gid);CHKERRQ(ierr);
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)
79d8185827SBarry Smith   err = getgroups(numGroups, gid+1); if (err < 0) SETERRQ(PETSC_COMM_SELF,PETSC_ERR_SYS, "Unable to obtain supplementary group IDs");
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   }
99e5c89e4eSSatish Balay   ierr = PetscFree(gid);CHKERRQ(ierr);
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   }
108e5c89e4eSSatish Balay   PetscFunctionReturn(0);
109e5c89e4eSSatish Balay }
110e5c89e4eSSatish Balay 
111e5c89e4eSSatish Balay #endif /* PETSC_HAVE_ACCESS */
112e5c89e4eSSatish Balay 
113ace3abfcSBarry Smith static PetscErrorCode PetscGetFileStat(const char fname[], uid_t *fileUid, gid_t *fileGid, int *fileMode,PetscBool  *exists)
114e5c89e4eSSatish Balay {
115e5c89e4eSSatish Balay   struct stat    statbuf;
116e5c89e4eSSatish Balay   PetscErrorCode 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)
1287fe8d12eSJed Brown     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");
1297ad82f04SSatish Balay #endif
1300298fd71SBarry Smith     ierr    = PetscInfo1(NULL,"System call stat() failed on file %s\n",fname);CHKERRQ(ierr);
131e5c89e4eSSatish Balay     *exists = PETSC_FALSE;
132e5c89e4eSSatish Balay   } else {
1330298fd71SBarry Smith     ierr      = PetscInfo1(NULL,"System call stat() succeeded on file %s\n",fname);CHKERRQ(ierr);
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   }
139e5c89e4eSSatish Balay   PetscFunctionReturn(0);
140e5c89e4eSSatish Balay }
141e5c89e4eSSatish Balay 
142e1d001d6SBarry Smith /*@C
14326b19047SBarry Smith    PetscTestFile - checks for the existence of a file
14426b19047SBarry Smith 
14526b19047SBarry Smith    Not Collective
14626b19047SBarry Smith 
147*d8d19677SJose 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 
15655819941SStefano Zampini    Notes: if mode is '\0', no permissions checks are performed
15755819941SStefano Zampini 
15826b19047SBarry Smith .seealso: PetscTestDirectory(), PetscLs()
15926b19047SBarry Smith @*/
1607087cfbeSBarry Smith PetscErrorCode  PetscTestFile(const char fname[], char mode, PetscBool  *flg)
161e5c89e4eSSatish Balay {
162e5c89e4eSSatish Balay   uid_t          fuid;
163e5c89e4eSSatish Balay   gid_t          fgid;
164e5c89e4eSSatish Balay   int            fmode;
165e5c89e4eSSatish Balay   PetscErrorCode ierr;
166ace3abfcSBarry Smith   PetscBool      exists;
167e5c89e4eSSatish Balay 
168e5c89e4eSSatish Balay   PetscFunctionBegin;
169e5c89e4eSSatish Balay   *flg = PETSC_FALSE;
170e5c89e4eSSatish Balay   if (!fname) PetscFunctionReturn(0);
171e5c89e4eSSatish Balay 
172e5c89e4eSSatish Balay   ierr = PetscGetFileStat(fname, &fuid, &fgid, &fmode, &exists);CHKERRQ(ierr);
173e5c89e4eSSatish Balay   if (!exists) PetscFunctionReturn(0);
1747fe8d12eSJed Brown   /* Except for systems that have this broken stat macros (rare), this is the correct way to check for a regular file */
175e5c89e4eSSatish Balay   if (!S_ISREG(fmode)) PetscFunctionReturn(0);
17655819941SStefano Zampini   /* return if asked to check for existence only */
17755819941SStefano Zampini   if (mode == '\0') { *flg = exists; PetscFunctionReturn(0); }
178e5c89e4eSSatish Balay   ierr = PetscTestOwnership(fname, mode, fuid, fgid, fmode, flg);CHKERRQ(ierr);
179e5c89e4eSSatish Balay   PetscFunctionReturn(0);
180e5c89e4eSSatish Balay }
181e5c89e4eSSatish Balay 
182e1d001d6SBarry Smith /*@C
18326b19047SBarry Smith    PetscTestDirectory - checks for the existence of a directory
18426b19047SBarry Smith 
18526b19047SBarry Smith    Not Collective
18626b19047SBarry Smith 
187*d8d19677SJose E. Roman    Input Parameters:
18826b19047SBarry Smith +  dirname - the directory name
18926b19047SBarry Smith -  mode - either 'r', 'w', or 'x'
19026b19047SBarry Smith 
19126b19047SBarry Smith    Output Parameter:
19226b19047SBarry Smith .  flg - the directory exists and satisfies the mode
19326b19047SBarry Smith 
1941b266c99SBarry Smith    Level: intermediate
1951b266c99SBarry Smith 
19626b19047SBarry Smith .seealso: PetscTestFile(), PetscLs()
19726b19047SBarry Smith @*/
19826b19047SBarry Smith PetscErrorCode  PetscTestDirectory(const char dirname[],char mode,PetscBool  *flg)
199e5c89e4eSSatish Balay {
200e5c89e4eSSatish Balay   uid_t          fuid;
201e5c89e4eSSatish Balay   gid_t          fgid;
202e5c89e4eSSatish Balay   int            fmode;
203e5c89e4eSSatish Balay   PetscErrorCode ierr;
204ace3abfcSBarry Smith   PetscBool      exists;
205e5c89e4eSSatish Balay 
206e5c89e4eSSatish Balay   PetscFunctionBegin;
207e5c89e4eSSatish Balay   *flg = PETSC_FALSE;
20826b19047SBarry Smith   if (!dirname) PetscFunctionReturn(0);
209e5c89e4eSSatish Balay 
21026b19047SBarry Smith   ierr = PetscGetFileStat(dirname, &fuid, &fgid, &fmode,&exists);CHKERRQ(ierr);
211e5c89e4eSSatish Balay   if (!exists) PetscFunctionReturn(0);
212e5c89e4eSSatish Balay   /* Except for systems that have this broken stat macros (rare), this
213e5c89e4eSSatish Balay      is the correct way to check for a directory */
214e5c89e4eSSatish Balay   if (!S_ISDIR(fmode)) PetscFunctionReturn(0);
215e5c89e4eSSatish Balay 
21626b19047SBarry Smith   ierr = PetscTestOwnership(dirname, mode, fuid, fgid, fmode, flg);CHKERRQ(ierr);
217e5c89e4eSSatish Balay   PetscFunctionReturn(0);
218e5c89e4eSSatish Balay }
219e5c89e4eSSatish Balay 
220e1d001d6SBarry Smith /*@C
22126b19047SBarry Smith    PetscLs - produce a listing of the files in a directory
22226b19047SBarry Smith 
223d083f849SBarry Smith    Collective
22426b19047SBarry Smith 
225*d8d19677SJose E. Roman    Input Parameters:
22626b19047SBarry Smith +  comm - the MPI communicator
22726b19047SBarry Smith .  dirname - the directory name
22826b19047SBarry Smith -  tlen - the length of the buffer found[]
22926b19047SBarry Smith 
230*d8d19677SJose E. Roman    Output Parameters:
23126b19047SBarry Smith +  found - listing of files
23226b19047SBarry Smith -  flg - the directory exists
23326b19047SBarry Smith 
2341b266c99SBarry Smith    Level: intermediate
2351b266c99SBarry Smith 
23626b19047SBarry Smith .seealso: PetscTestFile(), PetscLs()
23726b19047SBarry Smith @*/
23826b19047SBarry Smith PetscErrorCode  PetscLs(MPI_Comm comm,const char dirname[],char found[],size_t tlen,PetscBool  *flg)
239e5c89e4eSSatish Balay {
240e5c89e4eSSatish Balay   PetscErrorCode ierr;
241e5c89e4eSSatish Balay   size_t         len;
242e5c89e4eSSatish Balay   char           *f,program[PETSC_MAX_PATH_LEN];
243e5c89e4eSSatish Balay   FILE           *fp;
244e5c89e4eSSatish Balay 
245e5c89e4eSSatish Balay   PetscFunctionBegin;
246e5c89e4eSSatish Balay   ierr = PetscStrcpy(program,"ls ");CHKERRQ(ierr);
24726b19047SBarry Smith   ierr = PetscStrcat(program,dirname);CHKERRQ(ierr);
248e5c89e4eSSatish Balay #if defined(PETSC_HAVE_POPEN)
2490298fd71SBarry Smith   ierr = PetscPOpen(comm,NULL,program,"r",&fp);CHKERRQ(ierr);
250e5c89e4eSSatish Balay #else
251e32f2f54SBarry Smith   SETERRQ(PETSC_COMM_SELF,PETSC_ERR_SUP_SYS,"Cannot run external programs on this machine");
252e5c89e4eSSatish Balay #endif
253e5c89e4eSSatish Balay   f = fgets(found,tlen,fp);
254a297a907SKarl Rupp   if (f) *flg = PETSC_TRUE;
255a297a907SKarl Rupp   else *flg = PETSC_FALSE;
256e5c89e4eSSatish Balay   while (f) {
257e5c89e4eSSatish Balay     ierr = PetscStrlen(found,&len);CHKERRQ(ierr);
258e5c89e4eSSatish Balay     f    = fgets(found+len,tlen-len,fp);
259e5c89e4eSSatish Balay   }
26002c9f0b5SLisandro Dalcin   if (*flg) {ierr = PetscInfo2(NULL,"ls on %s gives \n%s\n",dirname,found);CHKERRQ(ierr);}
261e5c89e4eSSatish Balay #if defined(PETSC_HAVE_POPEN)
262016831caSBarry Smith   ierr = PetscPClose(comm,fp);CHKERRQ(ierr);
263e5c89e4eSSatish Balay #else
264e32f2f54SBarry Smith   SETERRQ(PETSC_COMM_SELF,PETSC_ERR_SUP_SYS,"Cannot run external programs on this machine");
265e5c89e4eSSatish Balay #endif
266e5c89e4eSSatish Balay   PetscFunctionReturn(0);
267e5c89e4eSSatish Balay }
268