Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/bin/pwd Make /bin/pwd almost conform to IEEE 1003.1



details:   https://anonhg.NetBSD.org/src/rev/ac7787f252ee
branches:  trunk
changeset: 554606:ac7787f252ee
user:      dsl <dsl%NetBSD.org@localhost>
date:      Thu Oct 30 13:52:23 2003 +0000

description:
Make /bin/pwd almost conform to IEEE 1003.1
- Make 'pwd -L' fall back to 'pwd -P' if PWD is incorrect.
- Ignore PWD if it contains "/./" or "/../".
- Garbage collect some redundant code.
It is still non-conformant because posix mandates that the default
be 'pwd -L' (aka ksh), not 'pwd -P' (historic practise everywhere else).
Changing the default will break too much...

diffstat:

 bin/pwd/pwd.1 |  43 ++++++++++++++++++++++++++-------
 bin/pwd/pwd.c |  75 +++++++++++++++++++++++++---------------------------------
 2 files changed, 66 insertions(+), 52 deletions(-)

diffs (220 lines):

diff -r 5d506ce170fe -r ac7787f252ee bin/pwd/pwd.1
--- a/bin/pwd/pwd.1     Thu Oct 30 13:18:24 2003 +0000
+++ b/bin/pwd/pwd.1     Thu Oct 30 13:52:23 2003 +0000
@@ -1,4 +1,4 @@
-.\"    $NetBSD: pwd.1,v 1.22 2003/08/07 09:05:25 agc Exp $
+.\"    $NetBSD: pwd.1,v 1.23 2003/10/30 13:52:23 dsl Exp $
 .\"
 .\" Copyright (c) 1990, 1993
 .\"    The Regents of the University of California.  All rights reserved.
@@ -32,7 +32,7 @@
 .\"
 .\"     @(#)pwd.1      8.2 (Berkeley) 4/28/95
 .\"
-.Dd November 2, 1998
+.Dd October 30, 2003
 .Dt PWD 1
 .Os
 .Sh NAME
@@ -49,15 +49,25 @@
 The following options are available:
 .Bl -tag -width indent
 .It Fl L
-Print the logical path to the current working directory, as indicated
-by the shell in the environment variable
-.Ev PWD ,
-if possible.
+If the
+.Ev PWD
+environment variable is an absolute pathname that contains
+neither "/./" nor "/../" and references the current directory, then
+.Ev PWD
+is assumed to be the name of the current directory.
 .It Fl P
 Print the physical path to the current working directory, with symbolic
 links in the path resolved.
-This is the default.
 .El
+.Pp
+The default for the
+.Nm
+command is
+.Fl P .
+.Pp
+.Nm
+is usually provided as a shell builtin (which may have a different
+default).
 .Sh EXIT STATUS
 The
 .Nm
@@ -65,13 +75,18 @@
 .Sh SEE ALSO
 .Xr cd 1 ,
 .Xr csh 1 ,
+.Xr ksh 1 ,
+.Xr sh 1 ,
 .Xr getcwd 3
 .Sh STANDARDS
 The
 .Nm
 utility is expected to be conforming to
-.St -p1003.2 .
-The options are extensions to the standard.
+.St -p1003.1 ,
+except that the default is
+.Fl P
+not
+.Fl L .
 .Sh BUGS
 In
 .Xr csh 1
@@ -80,3 +95,13 @@
 is always faster (although it can give a different answer in the rare case
 that the current directory or a containing directory was moved after
 the shell descended into it).
+.Pp
+.Nm
+.Fl L
+relies on the filesystem having unique inode numbers.
+If this is not true (eg msdos) then
+.Nm
+.Fl L
+may fail to detect that
+.Ev PWD
+is incorrect.
diff -r 5d506ce170fe -r ac7787f252ee bin/pwd/pwd.c
--- a/bin/pwd/pwd.c     Thu Oct 30 13:18:24 2003 +0000
+++ b/bin/pwd/pwd.c     Thu Oct 30 13:52:23 2003 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: pwd.c,v 1.18 2003/09/14 19:20:23 jschauma Exp $ */
+/* $NetBSD: pwd.c,v 1.19 2003/10/30 13:52:23 dsl Exp $ */
 
 /*
  * Copyright (c) 1991, 1993, 1994
@@ -39,7 +39,7 @@
 #if 0
 static char sccsid[] = "@(#)pwd.c      8.3 (Berkeley) 4/1/94";
 #else
-__RCSID("$NetBSD: pwd.c,v 1.18 2003/09/14 19:20:23 jschauma Exp $");
+__RCSID("$NetBSD: pwd.c,v 1.19 2003/10/30 13:52:23 dsl Exp $");
 #endif
 #endif /* not lint */
 
@@ -53,13 +53,17 @@
 #include <stdlib.h>
 #include <string.h>
 #include <unistd.h>
-#include <vis.h>
 
-int    stdout_ok;                      /* stdout connected to a terminal */
+static char *getcwd_logical(void);
+static void usage(void);
 
-static char *getcwd_logical(char *, size_t);
-static void usage(void);
-int main(int, char *[]);
+/*
+ * Note that EEE Std 1003.1, 2003 requires that the default be -L.
+ * This is inconsistent with the historic behaviour of everything
+ * except the ksh builtin.
+ * To avoid breaking scripts the default has been kept as -P.
+ * (Some scripts run /bin/pwd in order to get 'pwd -P'.)
+ */
 
 int
 main(int argc, char *argv[])
@@ -69,7 +73,7 @@
 
        setprogname(argv[0]);
        lFlag = 0;
-       while ((ch = getopt(argc, argv, "LP")) != -1)
+       while ((ch = getopt(argc, argv, "LP")) != -1) {
                switch (ch) {
                case 'L':
                        lFlag = 1;
@@ -81,6 +85,7 @@
                default:
                        usage();
                }
+       }
        argc -= optind;
        argv += optind;
 
@@ -88,15 +93,15 @@
                usage();
 
        if (lFlag)
-               p = getcwd_logical(NULL, 0);
+               p = getcwd_logical();
        else
+               p = NULL;
+       if (p == NULL)
                p = getcwd(NULL, 0);
 
        if (p == NULL)
                err(EXIT_FAILURE, NULL);
 
-       stdout_ok = isatty(STDOUT_FILENO);
-
        (void)printf("%s\n", p);
 
        exit(EXIT_SUCCESS);
@@ -104,42 +109,26 @@
 }
 
 static char *
-getcwd_logical(char *pt, size_t size)
+getcwd_logical(void)
 {
        char *pwd;
-       size_t pwdlen;
-       dev_t dev;
-       ino_t ino;
-       struct stat s;
+       struct stat s_pwd, s_dot;
 
        /* Check $PWD -- if it's right, it's fast. */
-       if ((pwd = getenv("PWD")) != NULL && pwd[0] == '/') {
-               if (stat(pwd, &s) != -1) {
-                       dev = s.st_dev;
-                       ino = s.st_ino;
-                       if (stat(".", &s) != -1 && dev == s.st_dev &&
-                           ino == s.st_ino) {
-                               pwdlen = strlen(pwd);
-                               if (pt) {
-                                       if (!size) {
-                                               errno = EINVAL;
-                                               return (NULL);
-                                       }
-                                       if (pwdlen + 1 > size) {
-                                               errno = ERANGE;
-                                               return (NULL);
-                                       }
-                               } else if ((pt = malloc(pwdlen + 1)) == NULL)
-                                       return (NULL);
-                               (void)memmove(pt, pwd, pwdlen);
-                               pt[pwdlen] = '\0';
-                               return (pt);
-                       }
-               }
-       } else
-               errno = ENOENT;
-
-       return (NULL);
+       pwd = getenv("PWD");
+       if (pwd == NULL)
+               return NULL;
+       if (pwd[0] != '/')
+               return NULL;
+       if (strstr(pwd, "/./") != NULL)
+               return NULL;
+       if (strstr(pwd, "/../") != NULL)
+               return NULL;
+       if (stat(pwd, &s_pwd) == -1 || stat(".", &s_dot) == -1)
+               return NULL;
+       if (s_pwd.st_dev != s_dot.st_dev || s_pwd.st_ino != s_dot.st_ino)
+               return NULL;
+       return pwd;
 }
 
 static void



Home | Main Index | Thread Index | Old Index