xref: /petsc/src/sys/fileio/ftest.c (revision 785e854f82a3c614b452fca2cf5ad4f2afe8bdde)
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 
24e5c89e4eSSatish Balay #undef __FUNCT__
25e5c89e4eSSatish Balay #define __FUNCT__ "PetscTestOwnership"
26ace3abfcSBarry Smith static PetscErrorCode PetscTestOwnership(const char fname[], char mode, uid_t fuid, gid_t fgid, int fmode, PetscBool  *flg)
27e5c89e4eSSatish Balay {
28e5c89e4eSSatish Balay   int            m = R_OK;
297fe8d12eSJed Brown   PetscErrorCode ierr;
30e5c89e4eSSatish Balay 
31e5c89e4eSSatish Balay   PetscFunctionBegin;
32e5c89e4eSSatish Balay   if (mode == 'r') m = R_OK;
33e5c89e4eSSatish Balay   else if (mode == 'w') m = W_OK;
34e5c89e4eSSatish Balay   else if (mode == 'x') m = X_OK;
35e32f2f54SBarry Smith   else SETERRQ(PETSC_COMM_SELF,PETSC_ERR_ARG_WRONG, "Mode must be one of r, w, or x");
36e5c89e4eSSatish Balay #if defined(PETSC_HAVE_ACCESS)
377fe8d12eSJed Brown   if (!access(fname, m)) {
380298fd71SBarry Smith     ierr = PetscInfo1(NULL,"System call access() succeeded on file %s\n",fname);CHKERRQ(ierr);
397fe8d12eSJed Brown     *flg = PETSC_TRUE;
407fe8d12eSJed Brown   } else {
410298fd71SBarry Smith     ierr = PetscInfo1(NULL,"System call access() failed on file %s\n",fname);CHKERRQ(ierr);
427fe8d12eSJed Brown     *flg = PETSC_FALSE;
437fe8d12eSJed Brown   }
44e5c89e4eSSatish Balay #else
45e32f2f54SBarry Smith   if (m == X_OK) SETERRQ1(PETSC_COMM_SELF,PETSC_ERR_SUP, "Unable to check execute permission for file %s", fname);
46e5c89e4eSSatish Balay   if (!_access(fname, m)) *flg = PETSC_TRUE;
47e5c89e4eSSatish Balay #endif
48e5c89e4eSSatish Balay   PetscFunctionReturn(0);
49e5c89e4eSSatish Balay }
50e5c89e4eSSatish Balay 
51e5c89e4eSSatish Balay #else  /* PETSC_HAVE_ACCESS or PETSC_HAVE__ACCESS */
52e5c89e4eSSatish Balay 
53e5c89e4eSSatish Balay #undef __FUNCT__
54e5c89e4eSSatish Balay #define __FUNCT__ "PetscTestOwnership"
55ace3abfcSBarry Smith static PetscErrorCode PetscTestOwnership(const char fname[], char mode, uid_t fuid, gid_t fgid, int fmode, PetscBool  *flg)
56e5c89e4eSSatish Balay {
57e5c89e4eSSatish Balay   uid_t          uid;
580298fd71SBarry Smith   gid_t          *gid = NULL;
59e5c89e4eSSatish Balay   int            numGroups;
60e5c89e4eSSatish Balay   int            rbit = S_IROTH;
61e5c89e4eSSatish Balay   int            wbit = S_IWOTH;
62e5c89e4eSSatish Balay   int            ebit = S_IXOTH;
63e5c89e4eSSatish Balay   PetscErrorCode ierr;
64e5c89e4eSSatish Balay 
65e5c89e4eSSatish Balay   PetscFunctionBegin;
66e5c89e4eSSatish Balay   /* Get the number of supplementary group IDs */
67e5c89e4eSSatish Balay #if !defined(PETSC_MISSING_GETGROUPS)
68c1235816SBarry Smith   numGroups = getgroups(0, gid); if (numGroups < 0) SETERRQ(PETSC_COMM_SELF,numGroups, "Unable to count supplementary group IDs");
69*785e854fSJed Brown   ierr = PetscMalloc1((numGroups+1), &gid);CHKERRQ(ierr);
70e5c89e4eSSatish Balay #else
71e5c89e4eSSatish Balay   numGroups = 0;
72e5c89e4eSSatish Balay #endif
73e5c89e4eSSatish Balay 
74e5c89e4eSSatish Balay   /* Get the (effective) user and group of the caller */
75e5c89e4eSSatish Balay   uid    = geteuid();
76e5c89e4eSSatish Balay   gid[0] = getegid();
77e5c89e4eSSatish Balay 
78e5c89e4eSSatish Balay   /* Get supplementary group IDs */
79e5c89e4eSSatish Balay #if !defined(PETSC_MISSING_GETGROUPS)
80c1235816SBarry Smith   ierr = getgroups(numGroups, gid+1); if (ierr < 0) SETERRQ(PETSC_COMM_SELF,ierr, "Unable to obtain supplementary group IDs");
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   }
100e5c89e4eSSatish Balay   ierr = PetscFree(gid);CHKERRQ(ierr);
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   }
109e5c89e4eSSatish Balay   PetscFunctionReturn(0);
110e5c89e4eSSatish Balay }
111e5c89e4eSSatish Balay 
112e5c89e4eSSatish Balay #endif /* PETSC_HAVE_ACCESS */
113e5c89e4eSSatish Balay 
114e5c89e4eSSatish Balay #undef __FUNCT__
115e5c89e4eSSatish Balay #define __FUNCT__ "PetscGetFileStat"
116ace3abfcSBarry Smith static PetscErrorCode PetscGetFileStat(const char fname[], uid_t *fileUid, gid_t *fileGid, int *fileMode,PetscBool  *exists)
117e5c89e4eSSatish Balay {
118e5c89e4eSSatish Balay   struct stat    statbuf;
119e5c89e4eSSatish Balay   PetscErrorCode ierr;
120e5c89e4eSSatish Balay 
121e5c89e4eSSatish Balay   PetscFunctionBegin;
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)
1297fe8d12eSJed 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");
1307ad82f04SSatish Balay #endif
1310298fd71SBarry Smith     ierr    = PetscInfo1(NULL,"System call stat() failed on file %s\n",fname);CHKERRQ(ierr);
132e5c89e4eSSatish Balay     *exists = PETSC_FALSE;
133e5c89e4eSSatish Balay   } else {
1340298fd71SBarry Smith     ierr      = PetscInfo1(NULL,"System call stat() succeeded on file %s\n",fname);CHKERRQ(ierr);
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   }
140e5c89e4eSSatish Balay   PetscFunctionReturn(0);
141e5c89e4eSSatish Balay }
142e5c89e4eSSatish Balay 
143e5c89e4eSSatish Balay #undef __FUNCT__
144e5c89e4eSSatish Balay #define __FUNCT__ "PetscTestFile"
1457087cfbeSBarry Smith PetscErrorCode  PetscTestFile(const char fname[], char mode, PetscBool  *flg)
146e5c89e4eSSatish Balay {
147e5c89e4eSSatish Balay   uid_t          fuid;
148e5c89e4eSSatish Balay   gid_t          fgid;
149e5c89e4eSSatish Balay   int            fmode;
150e5c89e4eSSatish Balay   PetscErrorCode ierr;
151ace3abfcSBarry Smith   PetscBool      exists;
152e5c89e4eSSatish Balay 
153e5c89e4eSSatish Balay   PetscFunctionBegin;
154e5c89e4eSSatish Balay   *flg = PETSC_FALSE;
155e5c89e4eSSatish Balay   if (!fname) PetscFunctionReturn(0);
156e5c89e4eSSatish Balay 
157e5c89e4eSSatish Balay   ierr = PetscGetFileStat(fname, &fuid, &fgid, &fmode,&exists);CHKERRQ(ierr);
158e5c89e4eSSatish Balay   if (!exists) PetscFunctionReturn(0);
1597fe8d12eSJed Brown   /* Except for systems that have this broken stat macros (rare), this is the correct way to check for a regular file */
160e5c89e4eSSatish Balay   if (!S_ISREG(fmode)) PetscFunctionReturn(0);
161e5c89e4eSSatish Balay 
162e5c89e4eSSatish Balay   ierr = PetscTestOwnership(fname, mode, fuid, fgid, fmode, flg);CHKERRQ(ierr);
163e5c89e4eSSatish Balay   PetscFunctionReturn(0);
164e5c89e4eSSatish Balay }
165e5c89e4eSSatish Balay 
166e5c89e4eSSatish Balay #undef __FUNCT__
167e5c89e4eSSatish Balay #define __FUNCT__ "PetscTestDirectory"
1687087cfbeSBarry Smith PetscErrorCode  PetscTestDirectory(const char fname[],char mode,PetscBool  *flg)
169e5c89e4eSSatish Balay {
170e5c89e4eSSatish Balay   uid_t          fuid;
171e5c89e4eSSatish Balay   gid_t          fgid;
172e5c89e4eSSatish Balay   int            fmode;
173e5c89e4eSSatish Balay   PetscErrorCode ierr;
174ace3abfcSBarry Smith   PetscBool      exists;
175e5c89e4eSSatish Balay 
176e5c89e4eSSatish Balay   PetscFunctionBegin;
177e5c89e4eSSatish Balay   *flg = PETSC_FALSE;
178e5c89e4eSSatish Balay   if (!fname) PetscFunctionReturn(0);
179e5c89e4eSSatish Balay 
180e5c89e4eSSatish Balay   ierr = PetscGetFileStat(fname, &fuid, &fgid, &fmode,&exists);CHKERRQ(ierr);
181e5c89e4eSSatish Balay   if (!exists) PetscFunctionReturn(0);
182e5c89e4eSSatish Balay   /* Except for systems that have this broken stat macros (rare), this
183e5c89e4eSSatish Balay      is the correct way to check for a directory */
184e5c89e4eSSatish Balay   if (!S_ISDIR(fmode)) PetscFunctionReturn(0);
185e5c89e4eSSatish Balay 
186e5c89e4eSSatish Balay   ierr = PetscTestOwnership(fname, mode, fuid, fgid, fmode, flg);CHKERRQ(ierr);
187e5c89e4eSSatish Balay   PetscFunctionReturn(0);
188e5c89e4eSSatish Balay }
189e5c89e4eSSatish Balay 
190e5c89e4eSSatish Balay #undef __FUNCT__
191e5c89e4eSSatish Balay #define __FUNCT__ "PetscLs"
1927087cfbeSBarry Smith PetscErrorCode  PetscLs(MPI_Comm comm,const char libname[],char found[],size_t tlen,PetscBool  *flg)
193e5c89e4eSSatish Balay {
194e5c89e4eSSatish Balay   PetscErrorCode ierr;
195e5c89e4eSSatish Balay   size_t         len;
196e5c89e4eSSatish Balay   char           *f,program[PETSC_MAX_PATH_LEN];
197e5c89e4eSSatish Balay   FILE           *fp;
198e5c89e4eSSatish Balay 
199e5c89e4eSSatish Balay   PetscFunctionBegin;
200e5c89e4eSSatish Balay   ierr = PetscStrcpy(program,"ls ");CHKERRQ(ierr);
201e5c89e4eSSatish Balay   ierr = PetscStrcat(program,libname);CHKERRQ(ierr);
202e5c89e4eSSatish Balay #if defined(PETSC_HAVE_POPEN)
2030298fd71SBarry Smith   ierr = PetscPOpen(comm,NULL,program,"r",&fp);CHKERRQ(ierr);
204e5c89e4eSSatish Balay #else
205e32f2f54SBarry Smith   SETERRQ(PETSC_COMM_SELF,PETSC_ERR_SUP_SYS,"Cannot run external programs on this machine");
206e5c89e4eSSatish Balay #endif
207e5c89e4eSSatish Balay   f = fgets(found,tlen,fp);
208a297a907SKarl Rupp   if (f) *flg = PETSC_TRUE;
209a297a907SKarl Rupp   else *flg = PETSC_FALSE;
210e5c89e4eSSatish Balay   while (f) {
211e5c89e4eSSatish Balay     ierr = PetscStrlen(found,&len);CHKERRQ(ierr);
212e5c89e4eSSatish Balay     f    = fgets(found+len,tlen-len,fp);
213e5c89e4eSSatish Balay   }
214ae15b995SBarry Smith   if (*flg) {ierr = PetscInfo2(0,"ls on %s gives \n%s\n",libname,found);CHKERRQ(ierr);}
215e5c89e4eSSatish Balay #if defined(PETSC_HAVE_POPEN)
2160298fd71SBarry Smith   ierr = PetscPClose(comm,fp,NULL);CHKERRQ(ierr);
217e5c89e4eSSatish Balay #else
218e32f2f54SBarry Smith   SETERRQ(PETSC_COMM_SELF,PETSC_ERR_SUP_SYS,"Cannot run external programs on this machine");
219e5c89e4eSSatish Balay #endif
220e5c89e4eSSatish Balay   PetscFunctionReturn(0);
221e5c89e4eSSatish Balay }
222