2 * Copyright (c) 1983, 1992, 1993
3 * The Regents of the University of California. All rights reserved.
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * 4. Neither the name of the University nor the names of its contributors
14 * may be used to endorse or promote products derived from this software
15 * without specific prior written permission.
17 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
18 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
21 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32 static char const copyright
[] =
33 "@(#) Copyright (c) 1983, 1992, 1993\n\
34 The Regents of the University of California. All rights reserved.\n";
38 static char sccsid
[] = "@(#)mkdir.c 8.2 (Berkeley) 1/25/94";
40 #include <sys/cdefs.h>
41 __FBSDID("$FreeBSD: src/bin/mkdir/mkdir.c,v 1.28 2004/04/06 20:06:48 markm Exp $");
44 #include <sys/types.h>
60 # include "mscfakes.h"
62 #include "kmkbuiltin.h"
66 static struct option long_options
[] =
68 { "help", no_argument
, 0, 261 },
69 { "version", no_argument
, 0, 262 },
74 extern void * bsd_setmode(const char *p
);
75 extern mode_t
bsd_getmode(const void *bbox
, mode_t omode
);
77 static int build(char *, mode_t
);
78 static int usage(FILE *);
82 kmk_builtin_mkdir(int argc
, char *argv
[], char **envp
)
84 int ch
, exitval
, success
, pflag
;
85 mode_t omode
, *set
= (mode_t
*)NULL
;
91 /* reinitialize globals */
94 /* kmk: reset getopt and set progname */
99 optind
= 0; /* init */
100 while ((ch
= getopt_long(argc
, argv
, "m:pv", long_options
, NULL
)) != -1)
115 return kbuild_version(argv
[0]);
118 return usage(stderr
);
124 return usage(stderr
);
127 omode
= S_IRWXU
| S_IRWXG
| S_IRWXO
;
129 if ((set
= bsd_setmode(mode
)) == NULL
)
130 return errx(1, "invalid file mode: %s", mode
);
131 omode
= bsd_getmode(set
, S_IRWXU
| S_IRWXG
| S_IRWXO
);
135 for (exitval
= 0; *argv
!= NULL
; ++argv
) {
138 if (build(*argv
, omode
))
140 } else if (mkdir(*argv
, omode
) < 0) {
141 if (errno
== ENOTDIR
|| errno
== ENOENT
)
142 warn("%s", dirname(*argv
));
147 (void)printf("%s\n", *argv
);
152 * The mkdir() and umask() calls both honor only the low
153 * nine bits, so if you try to set a mode including the
154 * sticky, setuid, setgid bits you lose them. Don't do
155 * this unless the user has specifically requested a mode,
156 * as chmod will (obviously) ignore the umask.
158 if (success
&& mode
!= NULL
&& chmod(*argv
, omode
) == -1) {
167 build(char *path
, mode_t omode
)
170 mode_t numask
, oumask
;
171 int first
, last
, retval
;
174 const size_t len
= strlen(path
);
176 path
= memcpy(p
, path
, len
+ 1);
178 #if defined(_MSC_VER) || defined(__EMX__)
179 /* correct slashes. */
180 p
= strchr(path
, '\\');
190 #if defined(_MSC_VER) || defined(__EMX__)
191 if ( ( (p
[0] >= 'A' && p
[0] <= 'Z')
192 || (p
[0] >= 'a' && p
[0] <= 'z'))
194 p
+= 2; /* skip the drive letter */
195 else if ( p
[0] == '/'
201 while (*p
&& *p
!= '/') /* server */
205 while (*p
&& *p
!= '/') /* share */
209 if (p
[0] == '/') /* Skip leading '/'. */
211 for (first
= 1, last
= 0; !last
; ++p
) {
214 else if (p
[0] != '/')
217 if (!last
&& p
[1] == '\0')
222 * For each dir operand that does not name an existing
223 * directory, effects equivalent to those cased by the
224 * following command shall occcur:
226 * mkdir -p -m $(umask -S),u+wx $(dirname dir) &&
227 * mkdir [-m mode] dir
229 * We change the user's umask and then restore it,
230 * instead of doing chmod's.
233 numask
= oumask
& ~(S_IWUSR
| S_IXUSR
);
239 if (mkdir(path
, last
? omode
: S_IRWXU
| S_IRWXG
| S_IRWXO
) < 0) {
240 if (errno
== EEXIST
|| errno
== EISDIR
|| errno
== ENOSYS
/* (solaris crap) */) {
241 if (stat(path
, &sb
) < 0) {
245 } else if (!S_ISDIR(sb
.st_mode
)) {
260 printf("%s\n", path
);
272 fprintf(pf
, "usage: %s [-pv] [-m mode] directory ...\n"
274 " or: %s --version\n",
275 g_progname
, g_progname
, g_progname
);