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