70fdbdfb5660e62f413125df3e1b80663c7c5247
[platform/upstream/busybox.git] / coreutils / mkdir.c
1 /* vi: set sw=4 ts=4: */
2 /*
3  * Mini mkdir implementation for busybox
4  *
5  *
6  * Copyright (C) 1999 by Lineo, inc.
7  * Written by Erik Andersen <andersen@lineo.com>, <andersee@debian.org>
8  *
9  * This program is free software; you can redistribute it and/or modify
10  * it under the terms of the GNU General Public License as published by
11  * the Free Software Foundation; either version 2 of the License, or
12  * (at your option) any later version.
13  *
14  * This program is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17  * General Public License for more details.
18  *
19  * You should have received a copy of the GNU General Public License
20  * along with this program; if not, write to the Free Software
21  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22  *
23  */
24
25 #include "internal.h"
26 #define bb_need_name_too_long
27 #define BB_DECLARE_EXTERN
28 #include "messages.c"
29
30 #include <stdio.h>
31 #include <errno.h>
32 #include <sys/param.h>                  /* for PATH_MAX */
33
34 static const char mkdir_usage[] =
35         "mkdir [OPTION] DIRECTORY...\n\n"
36         "Create the DIRECTORY(ies), if they do not already exist\n\n"
37         "Options:\n"
38
39         "\t-m\tset permission mode (as in chmod), not rwxrwxrwx - umask\n"
40         "\t-p\tno error if existing, make parent directories as needed\n";
41
42
43 static int parentFlag = FALSE;
44 static mode_t mode = 0777;
45
46
47 extern int mkdir_main(int argc, char **argv)
48 {
49         int i = FALSE;
50
51         argc--;
52         argv++;
53
54         /* Parse any options */
55         while (argc > 0 && **argv == '-') {
56                 while (i == FALSE && *++(*argv)) {
57                         switch (**argv) {
58                         case 'm':
59                                 if (--argc == 0)
60                                         usage(mkdir_usage);
61                                 /* Find the specified modes */
62                                 mode = 0;
63                                 if (parse_mode(*(++argv), &mode) == FALSE) {
64                                         fprintf(stderr, "Unknown mode: %s\n", *argv);
65                                         exit FALSE;
66                                 }
67                                 /* Set the umask for this process so it doesn't 
68                                  * screw up whatever the user just entered. */
69                                 umask(0);
70                                 i = TRUE;
71                                 break;
72                         case 'p':
73                                 parentFlag = TRUE;
74                                 break;
75                         default:
76                                 usage(mkdir_usage);
77                         }
78                 }
79                 argc--;
80                 argv++;
81         }
82
83         if (argc < 1) {
84                 usage(mkdir_usage);
85         }
86
87         while (argc > 0) {
88                 int status;
89                 struct stat statBuf;
90                 char buf[PATH_MAX + 1];
91
92                 if (strlen(*argv) > PATH_MAX - 1) {
93                         fprintf(stderr, name_too_long, "mkdir");
94                         exit FALSE;
95                 }
96                 strcpy(buf, *argv);
97                 status = stat(buf, &statBuf);
98                 if (parentFlag == FALSE && status != -1 && errno != ENOENT) {
99                         fprintf(stderr, "%s: File exists\n", buf);
100                         exit FALSE;
101                 }
102                 if (parentFlag == TRUE) {
103                         strcat(buf, "/");
104                         createPath(buf, mode);
105                 } else {
106                         if (mkdir(buf, mode) != 0 && parentFlag == FALSE) {
107                                 perror(buf);
108                                 exit FALSE;
109                         }
110                 }
111                 argc--;
112                 argv++;
113         }
114         exit TRUE;
115 }