rename to gl_FUNC_XFTS
[platform/upstream/coreutils.git] / src / mkdir.c
index 5ed40b8..a93572a 100644 (file)
@@ -1,5 +1,5 @@
 /* mkdir -- make directories
-   Copyright (C) 1990 Free Software Foundation, Inc.
+   Copyright (C) 90, 1995-2002, 2004, 2005 Free Software Foundation, Inc.
 
    This program is free software; you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
    GNU General Public License for more details.
 
    You should have received a copy of the GNU General Public License
-   along with this program; if not, write to the Free Software
-   Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.  */
-
-/* Options:
-   -p, --parent                Ensure that the given path(s) exist:
-                       Make any missing parent directories for each argument.
-                       Parent dirs default to umask modified by `u+wx'.
-                       Do not consider an argument directory that already
-                       exists to be an error.
-   -m, --mode=mode     Set the mode of created directories to `mode', which is
-                       symbolic as in chmod and uses the umask as a point of
-                       departure.
-
-   David MacKenzie <djm@ai.mit.edu>  */
-
-#ifdef HAVE_CONFIG_H
-#if defined (CONFIG_BROKETS)
-/* We use <config.h> instead of "config.h" so that a compilation
-   using -I. -I$srcdir will use ./config.h rather than $srcdir/config.h
-   (which it would do because it found this file in $srcdir).  */
-#include <config.h>
-#else
-#include "config.h"
-#endif
-#endif
+   along with this program; if not, write to the Free Software Foundation,
+   Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.  */
+
+/* David MacKenzie <djm@ai.mit.edu>  */
 
+#include <config.h>
 #include <stdio.h>
 #include <getopt.h>
 #include <sys/types.h>
+
 #include "system.h"
+#include "dirname.h"
+#include "error.h"
+#include "mkdir-p.h"
 #include "modechange.h"
-#include "version.h"
+#include "quote.h"
 
-int make_path ();
-void error ();
+/* The official name of this program (e.g., no `g' prefix).  */
+#define PROGRAM_NAME "mkdir"
 
-static void usage ();
+#define AUTHORS "David MacKenzie"
 
 /* The name this program was run with. */
 char *program_name;
 
-/* If nonzero, ensure that all parents of the specified directory exist.  */
-static int path_mode;
-
-/* If non-zero, display usage information and exit.  */
-static int show_help;
-
-/* If non-zero, print the version on standard output and exit.  */
-static int show_version;
-
 static struct option const longopts[] =
 {
   {"mode", required_argument, NULL, 'm'},
-  {"path", no_argument, &path_mode, 1},
-  {"parents", no_argument, &path_mode, 1},
-  {"help", no_argument, &show_help, 1},
-  {"version", no_argument, &show_version, 1},
+  {"parents", no_argument, NULL, 'p'},
+  {"verbose", no_argument, NULL, 'v'},
+  {GETOPT_HELP_OPTION_DECL},
+  {GETOPT_VERSION_OPTION_DECL},
   {NULL, 0, NULL, 0}
 };
 
 void
-main (argc, argv)
-     int argc;
-     char **argv;
+usage (int status)
 {
-  unsigned int newmode;
-  unsigned int parent_mode;
-  char *symbolic_mode = NULL;
-  int errors = 0;
+  if (status != EXIT_SUCCESS)
+    fprintf (stderr, _("Try `%s --help' for more information.\n"),
+            program_name);
+  else
+    {
+      printf (_("Usage: %s [OPTION] DIRECTORY...\n"), program_name);
+      fputs (_("\
+Create the DIRECTORY(ies), if they do not already exist.\n\
+\n\
+"), stdout);
+      fputs (_("\
+Mandatory arguments to long options are mandatory for short options too.\n\
+"), stdout);
+      fputs (_("\
+  -m, --mode=MODE   set permission mode (as in chmod), not rwxrwxrwx - umask\n\
+  -p, --parents     no error if existing, make parent directories as needed\n\
+  -v, --verbose     print a message for each created directory\n\
+"), stdout);
+      fputs (HELP_OPTION_DESCRIPTION, stdout);
+      fputs (VERSION_OPTION_DESCRIPTION, stdout);
+      printf (_("\nReport bugs to <%s>.\n"), PACKAGE_BUGREPORT);
+    }
+  exit (status);
+}
+
+int
+main (int argc, char **argv)
+{
+  mode_t newmode;
+  mode_t parent_mode IF_LINT (= 0);
+  const char *specified_mode = NULL;
+  const char *verbose_fmt_string = NULL;
+  bool create_parents = false;
+  int exit_status = EXIT_SUCCESS;
   int optc;
+  int cwd_errno = 0;
 
+  initialize_main (&argc, &argv);
   program_name = argv[0];
-  path_mode = 0;
+  setlocale (LC_ALL, "");
+  bindtextdomain (PACKAGE, LOCALEDIR);
+  textdomain (PACKAGE);
 
-  while ((optc = getopt_long (argc, argv, "pm:", longopts, (int *) 0)) != EOF)
+  atexit (close_stdout);
+
+  while ((optc = getopt_long (argc, argv, "pm:v", longopts, NULL)) != -1)
     {
       switch (optc)
        {
-       case 0:                 /* Long option. */
-         break;
        case 'p':
-         path_mode = 1;
+         create_parents = true;
          break;
        case 'm':
-         symbolic_mode = optarg;
+         specified_mode = optarg;
+         break;
+       case 'v': /* --verbose  */
+         verbose_fmt_string = _("created directory %s");
          break;
+       case_GETOPT_HELP_CHAR;
+       case_GETOPT_VERSION_CHAR (PROGRAM_NAME, AUTHORS);
        default:
-         usage (1);
+         usage (EXIT_FAILURE);
        }
     }
 
-  if (show_version)
+  if (optind == argc)
     {
-      printf ("%s\n", version_string);
-      exit (0);
+      error (0, 0, _("missing operand"));
+      usage (EXIT_FAILURE);
     }
 
-  if (show_help)
-    usage (0);
+  newmode = S_IRWXUGO;
 
-  if (optind == argc)
-    usage (1);
-
-  newmode = 0777 & ~umask (0);
-  parent_mode = newmode | 0300;        /* u+wx */
-  if (symbolic_mode)
+  if (specified_mode || create_parents)
     {
-      struct mode_change *change = mode_compile (symbolic_mode, 0);
-      if (change == MODE_INVALID)
-       error (1, 0, "invalid mode `%s'", symbolic_mode);
-      else if (change == MODE_MEMORY_EXHAUSTED)
-       error (1, 0, "virtual memory exhausted");
-      newmode = mode_adjust (newmode, change);
-    }
+      mode_t umask_value = umask (0);
 
-  for (; optind < argc; ++optind)
-    {
-      if (path_mode)
-       errors |= make_path (argv[optind], newmode, parent_mode, -1, -1, NULL);
-      else if (mkdir (argv[optind], newmode))
+      parent_mode = (S_IRWXUGO & ~umask_value) | (S_IWUSR | S_IXUSR);
+
+      if (specified_mode)
        {
-         error (0, errno, "cannot make directory `%s'", argv[optind]);
-         errors = 1;
+         struct mode_change *change = mode_compile (specified_mode);
+         if (!change)
+           error (EXIT_FAILURE, 0, _("invalid mode %s"),
+                  quote (specified_mode));
+         newmode = mode_adjust (S_IRWXUGO, change, umask_value);
+         free (change);
        }
+      else
+       umask (umask_value);
     }
 
-  exit (errors);
-}
-
-static void
-usage (status)
-     int status;
-{
-  fprintf (stderr, "\
-Usage: %s [OPTION] DIRECTORY...\n\
-\n",
-          program_name);
+  for (; optind < argc; ++optind)
+    {
+      char *dir = argv[optind];
+      bool ok;
 
-  if (status == 0)
-    fprintf (stderr, "\
-  -p, --parents     no error if existing, make parent directories as needed\n\
-  -m, --mode MODE   set permission mode (as in chmod), not 0777 - umask\n\
-      --help        provide this help\n\
-      --version     show program version\n");
+      if (create_parents)
+       {
+         if (cwd_errno != 0 && IS_RELATIVE_FILE_NAME (dir))
+           {
+             error (0, cwd_errno, _("cannot return to working directory"));
+             ok = false;
+           }
+         else
+           ok = make_dir_parents (dir, newmode, parent_mode,
+                                  -1, -1, true, verbose_fmt_string,
+                                  &cwd_errno);
+       }
+      else
+       {
+         ok = (mkdir (dir, newmode) == 0);
+
+         if (! ok)
+           error (0, errno, _("cannot create directory %s"), quote (dir));
+         else if (verbose_fmt_string)
+           error (0, 0, verbose_fmt_string, quote (dir));
+
+         /* mkdir(2) is required to honor only the file permission bits.
+            In particular, it needn't do anything about `special' bits,
+            so if any were set in newmode, apply them with chmod.
+            This extra step is necessary in some cases when the containing
+            directory has a default ACL.  */
+
+         /* Set the permissions only if this directory has just
+            been created.  */
+
+         if (ok && specified_mode
+             && chmod (dir, newmode) != 0)
+           {
+             error (0, errno, _("cannot set permissions of directory %s"),
+                    quote (dir));
+             ok = false;
+           }
+       }
 
-  else
-    fprintf (stderr, "Try `%s --help' for more information.\n",
-            program_name);
+      if (! ok)
+       exit_status = EXIT_FAILURE;
+    }
 
-  exit (status);
+  exit (exit_status);
 }
-