xref: /titanic_50/usr/src/lib/libcmd/common/mkdir.c (revision da2e3ebdc1edfbc5028edf1354e7dd2fa69a7968)
1*da2e3ebdSchin /***********************************************************************
2*da2e3ebdSchin *                                                                      *
3*da2e3ebdSchin *               This software is part of the ast package               *
4*da2e3ebdSchin *           Copyright (c) 1992-2007 AT&T Knowledge Ventures            *
5*da2e3ebdSchin *                      and is licensed under the                       *
6*da2e3ebdSchin *                  Common Public License, Version 1.0                  *
7*da2e3ebdSchin *                      by AT&T Knowledge Ventures                      *
8*da2e3ebdSchin *                                                                      *
9*da2e3ebdSchin *                A copy of the License is available at                 *
10*da2e3ebdSchin *            http://www.opensource.org/licenses/cpl1.0.txt             *
11*da2e3ebdSchin *         (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9)         *
12*da2e3ebdSchin *                                                                      *
13*da2e3ebdSchin *              Information and Software Systems Research               *
14*da2e3ebdSchin *                            AT&T Research                             *
15*da2e3ebdSchin *                           Florham Park NJ                            *
16*da2e3ebdSchin *                                                                      *
17*da2e3ebdSchin *                 Glenn Fowler <gsf@research.att.com>                  *
18*da2e3ebdSchin *                  David Korn <dgk@research.att.com>                   *
19*da2e3ebdSchin *                                                                      *
20*da2e3ebdSchin ***********************************************************************/
21*da2e3ebdSchin #pragma prototyped
22*da2e3ebdSchin /*
23*da2e3ebdSchin  * David Korn
24*da2e3ebdSchin  * AT&T Bell Laboratories
25*da2e3ebdSchin  *
26*da2e3ebdSchin  * mkdir
27*da2e3ebdSchin  */
28*da2e3ebdSchin 
29*da2e3ebdSchin static const char usage[] =
30*da2e3ebdSchin "[-?\n@(#)$Id: mkdir (AT&T Research) 2006-08-27 $\n]"
31*da2e3ebdSchin USAGE_LICENSE
32*da2e3ebdSchin "[+NAME?mkdir - make directories]"
33*da2e3ebdSchin "[+DESCRIPTION?\bmkdir\b creates one or more directories.  By "
34*da2e3ebdSchin 	"default, the mode of created directories is \ba=rwx\b minus the "
35*da2e3ebdSchin 	"bits set in the \bumask\b(1).]"
36*da2e3ebdSchin "[m:mode]:[mode?Set the mode of created directories to \amode\a.  "
37*da2e3ebdSchin 	"\amode\a is symbolic or octal mode as in \bchmod\b(1).  Relative "
38*da2e3ebdSchin 	"modes assume an initial mode of \ba=rwx\b.]"
39*da2e3ebdSchin "[p:parents?Create any missing intermediate pathname components. For "
40*da2e3ebdSchin     "each dir operand that does not name an existing directory, effects "
41*da2e3ebdSchin     "equivalent to those caused by the following command shall occur: "
42*da2e3ebdSchin     "\vmkdir -p -m $(umask -S),u+wx $(dirname dir) && mkdir [-m mode]] "
43*da2e3ebdSchin     "dir\v where the \b-m\b mode option represents that option supplied to "
44*da2e3ebdSchin     "the original invocation of \bmkdir\b, if any. Each dir operand that "
45*da2e3ebdSchin     "names an existing directory shall be ignored without error.]"
46*da2e3ebdSchin "\n"
47*da2e3ebdSchin "\ndirectory ...\n"
48*da2e3ebdSchin "\n"
49*da2e3ebdSchin "[+EXIT STATUS?]{"
50*da2e3ebdSchin         "[+0?All directories created successfully, or the \b-p\b option "
51*da2e3ebdSchin 	"was specified and all the specified directories now exist.]"
52*da2e3ebdSchin         "[+>0?An error occurred.]"
53*da2e3ebdSchin "}"
54*da2e3ebdSchin "[+SEE ALSO?\bchmod\b(1), \brmdir\b(1), \bumask\b(1)]"
55*da2e3ebdSchin ;
56*da2e3ebdSchin 
57*da2e3ebdSchin #include <cmd.h>
58*da2e3ebdSchin #include <ls.h>
59*da2e3ebdSchin 
60*da2e3ebdSchin #define DIRMODE	(S_IRWXU|S_IRWXG|S_IRWXO)
61*da2e3ebdSchin 
62*da2e3ebdSchin int
63*da2e3ebdSchin b_mkdir(int argc, char** argv, void* context)
64*da2e3ebdSchin {
65*da2e3ebdSchin 	register char*	arg;
66*da2e3ebdSchin 	register int	n;
67*da2e3ebdSchin 	register mode_t	mode = DIRMODE;
68*da2e3ebdSchin 	register mode_t	mask = 0;
69*da2e3ebdSchin 	register int	mflag = 0;
70*da2e3ebdSchin 	register int	pflag = 0;
71*da2e3ebdSchin 	char*		name;
72*da2e3ebdSchin 	mode_t		dmode;
73*da2e3ebdSchin 
74*da2e3ebdSchin 	cmdinit(argc, argv, context, ERROR_CATALOG, 0);
75*da2e3ebdSchin 	while (n = optget(argv, usage)) switch (n)
76*da2e3ebdSchin 	{
77*da2e3ebdSchin 	case 'p':
78*da2e3ebdSchin 		pflag = 1;
79*da2e3ebdSchin 		break;
80*da2e3ebdSchin 	case 'm':
81*da2e3ebdSchin 		mflag = 1;
82*da2e3ebdSchin 		mode = strperm(arg = opt_info.arg, &opt_info.arg, mode);
83*da2e3ebdSchin 		if (*opt_info.arg)
84*da2e3ebdSchin 			error(ERROR_exit(0), "%s: invalid mode", arg);
85*da2e3ebdSchin 		break;
86*da2e3ebdSchin 	case ':':
87*da2e3ebdSchin 		error(2, "%s", opt_info.arg);
88*da2e3ebdSchin 		break;
89*da2e3ebdSchin 	case '?':
90*da2e3ebdSchin 		error(ERROR_usage(2), "%s", opt_info.arg);
91*da2e3ebdSchin 		break;
92*da2e3ebdSchin 	}
93*da2e3ebdSchin 	argv += opt_info.index;
94*da2e3ebdSchin 	if (error_info.errors || !*argv)
95*da2e3ebdSchin 		error(ERROR_usage(2), "%s", optusage(NiL));
96*da2e3ebdSchin 	mask = umask(0);
97*da2e3ebdSchin 	if (mflag || pflag)
98*da2e3ebdSchin 	{
99*da2e3ebdSchin 		dmode = DIRMODE & ~mask;
100*da2e3ebdSchin 		if (!mflag)
101*da2e3ebdSchin 			mode = dmode;
102*da2e3ebdSchin 		dmode |= S_IWUSR | S_IXUSR;
103*da2e3ebdSchin 	}
104*da2e3ebdSchin 	else
105*da2e3ebdSchin 	{
106*da2e3ebdSchin 		mode &= ~mask;
107*da2e3ebdSchin 		umask(mask);
108*da2e3ebdSchin 		mask = 0;
109*da2e3ebdSchin 	}
110*da2e3ebdSchin 	while (arg = *argv++)
111*da2e3ebdSchin 	{
112*da2e3ebdSchin 		if (mkdir(arg, mode) < 0)
113*da2e3ebdSchin 		{
114*da2e3ebdSchin 			if (!pflag || !(errno == ENOENT || errno == EEXIST || errno == ENOTDIR))
115*da2e3ebdSchin 			{
116*da2e3ebdSchin 				error(ERROR_system(0), "%s:", arg);
117*da2e3ebdSchin 				continue;
118*da2e3ebdSchin 			}
119*da2e3ebdSchin 			if (errno == EEXIST)
120*da2e3ebdSchin 				continue;
121*da2e3ebdSchin 
122*da2e3ebdSchin 			/*
123*da2e3ebdSchin 			 * -p option, preserve intermediates
124*da2e3ebdSchin 			 * first eliminate trailing /'s
125*da2e3ebdSchin 			 */
126*da2e3ebdSchin 
127*da2e3ebdSchin 			n = strlen(arg);
128*da2e3ebdSchin 			while (n > 0 && arg[--n] == '/');
129*da2e3ebdSchin 			arg[n + 1] = 0;
130*da2e3ebdSchin 			for (name = arg, n = *arg; n;)
131*da2e3ebdSchin 			{
132*da2e3ebdSchin 				/* skip over slashes */
133*da2e3ebdSchin 				while (*arg == '/')
134*da2e3ebdSchin 					arg++;
135*da2e3ebdSchin 				/* skip to next component */
136*da2e3ebdSchin 				while ((n = *arg) && n != '/')
137*da2e3ebdSchin 					arg++;
138*da2e3ebdSchin 				*arg = 0;
139*da2e3ebdSchin 				if (mkdir(name, n ? dmode : mode) < 0 && errno != EEXIST && access(name, F_OK) < 0)
140*da2e3ebdSchin 				{
141*da2e3ebdSchin 					*arg = n;
142*da2e3ebdSchin 					error(ERROR_system(0), "%s:", name);
143*da2e3ebdSchin 					break;
144*da2e3ebdSchin 				}
145*da2e3ebdSchin 				*arg = n;
146*da2e3ebdSchin 			}
147*da2e3ebdSchin 		}
148*da2e3ebdSchin 	}
149*da2e3ebdSchin 	if (mask)
150*da2e3ebdSchin 		umask(mask);
151*da2e3ebdSchin 	return error_info.errors != 0;
152*da2e3ebdSchin }
153