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