xref: /petsc/src/sys/fileio/ftest.c (revision ae15b995b5732fffd2de5a75cf61ef7190c6fef1)
1e5c89e4eSSatish Balay #define PETSC_DLL
2e5c89e4eSSatish Balay 
3e5c89e4eSSatish Balay #include "petsc.h"
4e5c89e4eSSatish Balay #include "petscsys.h"
5e5c89e4eSSatish Balay #if defined(PETSC_HAVE_PWD_H)
6e5c89e4eSSatish Balay #include <pwd.h>
7e5c89e4eSSatish Balay #endif
8e5c89e4eSSatish Balay #include <ctype.h>
9e5c89e4eSSatish Balay #include <sys/types.h>
10e5c89e4eSSatish Balay #include <sys/stat.h>
11e5c89e4eSSatish Balay #if defined(PETSC_HAVE_UNISTD_H)
12e5c89e4eSSatish Balay #include <unistd.h>
13e5c89e4eSSatish Balay #endif
14e5c89e4eSSatish Balay #if defined(PETSC_HAVE_STDLIB_H)
15e5c89e4eSSatish Balay #include <stdlib.h>
16e5c89e4eSSatish Balay #endif
17e5c89e4eSSatish Balay #if defined(PETSC_HAVE_SYS_UTSNAME_H)
18e5c89e4eSSatish Balay #include <sys/utsname.h>
19e5c89e4eSSatish Balay #endif
20e5c89e4eSSatish Balay #if defined(PETSC_HAVE_IO_H)
21e5c89e4eSSatish Balay #include <io.h>
22e5c89e4eSSatish Balay #endif
23e5c89e4eSSatish Balay #if defined(PETSC_HAVE_SYS_SYSTEMINFO_H)
24e5c89e4eSSatish Balay #include <sys/systeminfo.h>
25e5c89e4eSSatish Balay #endif
26e5c89e4eSSatish Balay #include "petscfix.h"
27e5c89e4eSSatish Balay 
28e5c89e4eSSatish Balay #if defined (PETSC_HAVE__ACCESS) || defined(PETSC_HAVE_ACCESS)
29e5c89e4eSSatish Balay 
30e5c89e4eSSatish Balay #undef __FUNCT__
31e5c89e4eSSatish Balay #define __FUNCT__ "PetscTestOwnership"
32e5c89e4eSSatish Balay static PetscErrorCode PetscTestOwnership(const char fname[], char mode, uid_t fuid, gid_t fgid, int fmode, PetscTruth *flg)
33e5c89e4eSSatish Balay {
34e5c89e4eSSatish Balay   int m = R_OK;
35e5c89e4eSSatish Balay 
36e5c89e4eSSatish Balay   PetscFunctionBegin;
37e5c89e4eSSatish Balay   if (mode == 'r') m = R_OK;
38e5c89e4eSSatish Balay   else if (mode == 'w') m = W_OK;
39e5c89e4eSSatish Balay   else if (mode == 'x') m = X_OK;
40e5c89e4eSSatish Balay   else SETERRQ(PETSC_ERR_ARG_WRONG, "Mode must be one of r, w, or x");
41e5c89e4eSSatish Balay #if defined(PETSC_HAVE_ACCESS)
42e5c89e4eSSatish Balay   if(!access(fname, m))  *flg = PETSC_TRUE;
43e5c89e4eSSatish Balay #else
44e5c89e4eSSatish Balay   if (m == X_OK) SETERRQ1(PETSC_ERR_SUP, "Unable to check execute permission for file %s", fname);
45e5c89e4eSSatish Balay   if(!_access(fname, m)) *flg = PETSC_TRUE;
46e5c89e4eSSatish Balay #endif
47e5c89e4eSSatish Balay   PetscFunctionReturn(0);
48e5c89e4eSSatish Balay }
49e5c89e4eSSatish Balay 
50e5c89e4eSSatish Balay #else  /* PETSC_HAVE_ACCESS or PETSC_HAVE__ACCESS */
51e5c89e4eSSatish Balay 
52e5c89e4eSSatish Balay #undef __FUNCT__
53e5c89e4eSSatish Balay #define __FUNCT__ "PetscTestOwnership"
54e5c89e4eSSatish Balay static PetscErrorCode PetscTestOwnership(const char fname[], char mode, uid_t fuid, gid_t fgid, int fmode, PetscTruth *flg)
55e5c89e4eSSatish Balay {
56e5c89e4eSSatish Balay   uid_t          uid;
57e5c89e4eSSatish Balay   gid_t          *gid = PETSC_NULL;
58e5c89e4eSSatish Balay   int            numGroups;
59e5c89e4eSSatish Balay   int            rbit = S_IROTH;
60e5c89e4eSSatish Balay   int            wbit = S_IWOTH;
61e5c89e4eSSatish Balay   int            ebit = S_IXOTH;
62e5c89e4eSSatish Balay   PetscErrorCode ierr;
63e5c89e4eSSatish Balay 
64e5c89e4eSSatish Balay   PetscFunctionBegin;
65e5c89e4eSSatish Balay   /* Get the number of supplementary group IDs */
66e5c89e4eSSatish Balay #if !defined(PETSC_MISSING_GETGROUPS)
67e5c89e4eSSatish Balay   numGroups = getgroups(0, gid); if (numGroups < 0) {SETERRQ(numGroups, "Unable to count supplementary group IDs");}
68e5c89e4eSSatish Balay   ierr = PetscMalloc((numGroups+1) * sizeof(gid_t), &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)
79e5c89e4eSSatish Balay   ierr = getgroups(numGroups, gid+1); if (ierr < 0) {SETERRQ(ierr, "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 
113e5c89e4eSSatish Balay #undef __FUNCT__
114e5c89e4eSSatish Balay #define __FUNCT__ "PetscGetFileStat"
115e5c89e4eSSatish Balay static PetscErrorCode PetscGetFileStat(const char fname[], uid_t *fileUid, gid_t *fileGid, int *fileMode,PetscTruth *exists)
116e5c89e4eSSatish Balay {
117e5c89e4eSSatish Balay   struct stat    statbuf;
118e5c89e4eSSatish Balay   PetscErrorCode ierr;
119e5c89e4eSSatish Balay 
120e5c89e4eSSatish Balay   PetscFunctionBegin;
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) {
127e5c89e4eSSatish Balay     *exists = PETSC_FALSE;
128e5c89e4eSSatish Balay   } else {
129e5c89e4eSSatish Balay     *exists = PETSC_TRUE;
130e5c89e4eSSatish Balay     *fileUid  = statbuf.st_uid;
131e5c89e4eSSatish Balay     *fileGid  = statbuf.st_gid;
132e5c89e4eSSatish Balay     *fileMode = statbuf.st_mode;
133e5c89e4eSSatish Balay   }
134e5c89e4eSSatish Balay   PetscFunctionReturn(0);
135e5c89e4eSSatish Balay }
136e5c89e4eSSatish Balay 
137e5c89e4eSSatish Balay #undef __FUNCT__
138e5c89e4eSSatish Balay #define __FUNCT__ "PetscTestFile"
139e5c89e4eSSatish Balay PetscErrorCode PETSC_DLLEXPORT PetscTestFile(const char fname[], char mode, PetscTruth *flg)
140e5c89e4eSSatish Balay {
141e5c89e4eSSatish Balay   uid_t          fuid;
142e5c89e4eSSatish Balay   gid_t          fgid;
143e5c89e4eSSatish Balay   int            fmode;
144e5c89e4eSSatish Balay   PetscErrorCode ierr;
145e5c89e4eSSatish Balay   PetscTruth     exists;
146e5c89e4eSSatish Balay 
147e5c89e4eSSatish Balay   PetscFunctionBegin;
148e5c89e4eSSatish Balay   *flg = PETSC_FALSE;
149e5c89e4eSSatish Balay   if (!fname) PetscFunctionReturn(0);
150e5c89e4eSSatish Balay 
151e5c89e4eSSatish Balay   ierr = PetscGetFileStat(fname, &fuid, &fgid, &fmode,&exists);CHKERRQ(ierr);
152e5c89e4eSSatish Balay   if (!exists) PetscFunctionReturn(0);
153e5c89e4eSSatish Balay   /* Except for systems that have this broken stat macros (rare), this
154e5c89e4eSSatish Balay      is the correct way to check for a regular file */
155e5c89e4eSSatish Balay   if (!S_ISREG(fmode)) PetscFunctionReturn(0);
156e5c89e4eSSatish Balay 
157e5c89e4eSSatish Balay   ierr = PetscTestOwnership(fname, mode, fuid, fgid, fmode, flg);CHKERRQ(ierr);
158e5c89e4eSSatish Balay   PetscFunctionReturn(0);
159e5c89e4eSSatish Balay }
160e5c89e4eSSatish Balay 
161e5c89e4eSSatish Balay #undef __FUNCT__
162e5c89e4eSSatish Balay #define __FUNCT__ "PetscTestDirectory"
163e5c89e4eSSatish Balay PetscErrorCode PETSC_DLLEXPORT PetscTestDirectory(const char fname[],char mode,PetscTruth *flg)
164e5c89e4eSSatish Balay {
165e5c89e4eSSatish Balay   uid_t          fuid;
166e5c89e4eSSatish Balay   gid_t          fgid;
167e5c89e4eSSatish Balay   int            fmode;
168e5c89e4eSSatish Balay   PetscErrorCode ierr;
169e5c89e4eSSatish Balay   PetscTruth     exists;
170e5c89e4eSSatish Balay 
171e5c89e4eSSatish Balay   PetscFunctionBegin;
172e5c89e4eSSatish Balay   *flg = PETSC_FALSE;
173e5c89e4eSSatish Balay   if (!fname) PetscFunctionReturn(0);
174e5c89e4eSSatish Balay 
175e5c89e4eSSatish Balay   ierr = PetscGetFileStat(fname, &fuid, &fgid, &fmode,&exists);CHKERRQ(ierr);
176e5c89e4eSSatish Balay   if (!exists) PetscFunctionReturn(0);
177e5c89e4eSSatish Balay   /* Except for systems that have this broken stat macros (rare), this
178e5c89e4eSSatish Balay      is the correct way to check for a directory */
179e5c89e4eSSatish Balay   if (!S_ISDIR(fmode)) PetscFunctionReturn(0);
180e5c89e4eSSatish Balay 
181e5c89e4eSSatish Balay   ierr = PetscTestOwnership(fname, mode, fuid, fgid, fmode, flg);CHKERRQ(ierr);
182e5c89e4eSSatish Balay   PetscFunctionReturn(0);
183e5c89e4eSSatish Balay }
184e5c89e4eSSatish Balay 
185e5c89e4eSSatish Balay #undef __FUNCT__
186e5c89e4eSSatish Balay #define __FUNCT__ "PetscLs"
187e5c89e4eSSatish Balay PetscErrorCode PETSC_DLLEXPORT PetscLs(MPI_Comm comm,const char libname[],char *found,size_t tlen,PetscTruth *flg)
188e5c89e4eSSatish Balay {
189e5c89e4eSSatish Balay   PetscErrorCode ierr;
190e5c89e4eSSatish Balay   size_t         len;
191e5c89e4eSSatish Balay   char           *f,program[PETSC_MAX_PATH_LEN];
192e5c89e4eSSatish Balay   FILE           *fp;
193e5c89e4eSSatish Balay 
194e5c89e4eSSatish Balay   PetscFunctionBegin;
195e5c89e4eSSatish Balay   ierr   = PetscStrcpy(program,"ls ");CHKERRQ(ierr);
196e5c89e4eSSatish Balay   ierr   = PetscStrcat(program,libname);CHKERRQ(ierr);
197e5c89e4eSSatish Balay #if defined(PETSC_HAVE_POPEN)
198e5c89e4eSSatish Balay   ierr   = PetscPOpen(comm,PETSC_NULL,program,"r",&fp);CHKERRQ(ierr);
199e5c89e4eSSatish Balay #else
200e5c89e4eSSatish Balay   SETERRQ(PETSC_ERR_SUP_SYS,"Cannot run external programs on this machine");
201e5c89e4eSSatish Balay #endif
202e5c89e4eSSatish Balay   f      = fgets(found,tlen,fp);
203e5c89e4eSSatish Balay   if (f) *flg = PETSC_TRUE; else *flg = PETSC_FALSE;
204e5c89e4eSSatish Balay   while (f) {
205e5c89e4eSSatish Balay     ierr  = PetscStrlen(found,&len);CHKERRQ(ierr);
206e5c89e4eSSatish Balay     f     = fgets(found+len,tlen-len,fp);
207e5c89e4eSSatish Balay   }
208*ae15b995SBarry Smith   if (*flg) {ierr = PetscInfo2(0,"ls on %s gives \n%s\n",libname,found);CHKERRQ(ierr);}
209e5c89e4eSSatish Balay #if defined(PETSC_HAVE_POPEN)
210e5c89e4eSSatish Balay   ierr   = PetscPClose(comm,fp);CHKERRQ(ierr);
211e5c89e4eSSatish Balay #else
212e5c89e4eSSatish Balay   SETERRQ(PETSC_ERR_SUP_SYS,"Cannot run external programs on this machine");
213e5c89e4eSSatish Balay #endif
214e5c89e4eSSatish Balay   PetscFunctionReturn(0);
215e5c89e4eSSatish Balay }
216