1 /* pathchk -- check whether pathnames are valid or portable
2 Copyright (C) 91, 92, 93, 94, 1995 Free Software Foundation, Inc.
4 This program is free software; you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation; either version 2, or (at your option)
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program; if not, write to the Free Software
16 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
18 /* Usage: pathchk [-p] [--portability] path...
20 For each PATH, print a message if any of these conditions are false:
21 * all existing leading directories in PATH have search (execute) permission
22 * strlen (PATH) <= PATH_MAX
23 * strlen (each_directory_in_PATH) <= NAME_MAX
26 0 All PATH names passed all of the tests.
30 -p, --portability Instead of performing length checks on the
31 underlying filesystem, test the length of the
32 pathname and its components against the POSIX.1
33 minimum limits for portability, _POSIX_NAME_MAX
34 and _POSIX_PATH_MAX in 2.9.2. Also check that
35 the pathname contains no character not in the
36 portable filename character set.
38 David MacKenzie <djm@gnu.ai.mit.edu>
39 and Jim Meyering <meyering@cs.utexas.edu> */
44 #include <sys/types.h>
53 #define PATH_MAX_FOR(p) pathconf ((p), _PC_PATH_MAX)
54 #endif /* not PATH_MAX */
56 #define NAME_MAX_FOR(p) pathconf ((p), _PC_NAME_MAX);
57 #endif /* not NAME_MAX */
59 #else /* not _POSIX_VERSION */
61 #include <sys/param.h>
64 #define PATH_MAX MAXPATHLEN
65 #else /* not MAXPATHLEN */
66 #define PATH_MAX _POSIX_PATH_MAX
67 #endif /* not MAXPATHLEN */
68 #endif /* not PATH_MAX */
72 #define NAME_MAX MAXNAMLEN
73 #else /* not MAXNAMLEN */
74 #define NAME_MAX _POSIX_NAME_MAX
75 #endif /* not MAXNAMLEN */
76 #endif /* not NAME_MAX */
78 #endif /* not _POSIX_VERSION */
80 #ifndef _POSIX_PATH_MAX
81 #define _POSIX_PATH_MAX 255
83 #ifndef _POSIX_NAME_MAX
84 #define _POSIX_NAME_MAX 14
88 #define PATH_MAX_FOR(p) PATH_MAX
91 #define NAME_MAX_FOR(p) NAME_MAX
96 static int validate_path __P ((char *path, int portability));
97 static void usage __P ((int status));
99 /* The name this program was run with. */
102 /* If nonzero, display usage information and exit. */
103 static int show_help;
105 /* If nonzero, print the version on standard output and exit. */
106 static int show_version;
108 static struct option const longopts[] =
110 {"help", no_argument, &show_help, 1},
111 {"portability", no_argument, NULL, 'p'},
112 {"version", no_argument, &show_version, 1},
117 main (int argc, char **argv)
120 int check_portability = 0;
123 program_name = argv[0];
124 setlocale (LC_ALL, "");
125 bindtextdomain (PACKAGE, LOCALEDIR);
126 textdomain (PACKAGE);
128 while ((optc = getopt_long (argc, argv, "p", longopts, (int *) 0)) != EOF)
136 check_portability = 1;
146 printf ("pathchk - %s\n", version_string);
155 error (0, 0, _("too few arguments"));
159 for (; optind < argc; ++optind)
160 exit_status |= validate_path (argv[optind], check_portability);
165 /* Each element is nonzero if the corresponding ASCII character is
166 in the POSIX portable character set, and zero if it is not.
167 In addition, the entry for `/' is nonzero to simplify checking. */
168 static char const portable_chars[256] =
170 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* 0-15 */
171 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* 16-31 */
172 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, /* 32-47 */
173 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, /* 48-63 */
174 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, /* 64-79 */
175 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 1, /* 80-95 */
176 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, /* 96-111 */
177 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, /* 112-127 */
178 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
179 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
180 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
181 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
182 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
183 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
184 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
185 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
188 /* If PATH contains only portable characters, return 1, else 0. */
191 portable_chars_only (const char *path)
195 for (p = path; *p; ++p)
196 if (portable_chars[(const unsigned char) *p] == 0)
198 error (0, 0, _("path `%s' contains nonportable character `%c'"),
205 /* Return 1 if PATH is a usable leading directory, 0 if not,
206 2 if it doesn't exist. */
209 dir_ok (const char *path)
213 if (stat (path, &stats))
216 if (!S_ISDIR (stats.st_mode))
218 error (0, 0, _("`%s' is not a directory"), path);
222 /* Use access to test for search permission because
223 testing permission bits of st_mode can lose with new
224 access control mechanisms. Of course, access loses if you're
226 if (access (path, X_OK) != 0)
229 error (0, 0, _("directory `%s' is not searchable"), path);
231 error (0, errno, "%s", path);
239 strlen (PATH) <= PATH_MAX
240 && strlen (each-existing-directory-in-PATH) <= NAME_MAX
242 If PORTABILITY is nonzero, compare against _POSIX_PATH_MAX and
243 _POSIX_NAME_MAX instead, and make sure that PATH contains no
244 characters not in the POSIX portable filename character set, which
245 consists of A-Z, a-z, 0-9, ., _, -.
247 Make sure that all leading directories along PATH that exist have
250 Return 0 if all of these tests are successful, 1 if any fail. */
253 validate_path (char *path, int portability)
256 int last_elem; /* Nonzero if checking last element of path. */
257 int exists; /* 2 if the path element exists. */
259 char *parent; /* Last existing leading directory so far. */
261 if (portability && !portable_chars_only (path))
268 /* Suppress `used before initialized' warning. */
272 /* Figure out the parent of the first element in PATH. */
273 parent = xstrdup (*path == '/' ? "/" : ".");
280 int length; /* Length of partial path being checked. */
281 char *start; /* Start of path element being checked. */
283 /* Find the end of this element of the path.
284 Then chop off the rest of the path after this element. */
285 while (*slash == '/')
288 slash = strchr (slash, '/');
294 slash = strchr (start, '\0');
299 exists = dir_ok (path);
307 length = slash - start;
308 /* Since we know that `parent' is a directory, it's ok to call
309 pathconf with it as the argument. (If `parent' isn't a directory
310 or doesn't exist, the behavior of pathconf is undefined.)
311 But if `parent' is a directory and is on a remote file system,
312 it's likely that pathconf can't give us a reasonable value
313 and will return -1. (NFS and tempfs are not POSIX . . .)
314 In that case, we have no choice but to assume the pessimal
316 name_max = portability ? _POSIX_NAME_MAX : NAME_MAX_FOR (parent);
318 name_max = _POSIX_NAME_MAX;
319 if (length > name_max)
321 error (0, 0, _("name `%s' has length %d; exceeds limit of %d"),
322 start, length, name_max);
333 parent = xstrdup (path);
339 /* `parent' is now the last existing leading directory in the whole path,
340 so it's ok to call pathconf with it as the argument. */
341 path_max = portability ? _POSIX_PATH_MAX : PATH_MAX_FOR (parent);
343 path_max = _POSIX_PATH_MAX;
345 if (strlen (path) > path_max)
347 error (0, 0, _("path `%s' has length %d; exceeds limit of %d"),
348 path, strlen (path), path_max);
359 fprintf (stderr, _("Try `%s --help' for more information.\n"),
363 printf (_("Usage: %s [OPTION]... NAME...\n"), program_name);
365 Diagnose unportable constructs in NAME.\n\
367 -p, --portability check for all POSIX systems, not only this one\n\
368 --help display this help and exit\n\
369 --version output version information and exit\n\