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 * Glenn Fowler 24*da2e3ebdSchin * AT&T Research 25*da2e3ebdSchin * 26*da2e3ebdSchin * rmdir 27*da2e3ebdSchin */ 28*da2e3ebdSchin 29*da2e3ebdSchin static const char usage[] = 30*da2e3ebdSchin "[-?\n@(#)$Id: rmdir (AT&T Research) 2006-08-24 $\n]" 31*da2e3ebdSchin USAGE_LICENSE 32*da2e3ebdSchin "[+NAME?rmdir - remove empty directories]" 33*da2e3ebdSchin "[+DESCRIPTION?\brmdir\b deletes each given directory. The directory " 34*da2e3ebdSchin "must be empty; containing no entries other than \b.\b or \b..\b. " 35*da2e3ebdSchin "If a directory and a subdirectory of that directory are specified " 36*da2e3ebdSchin "as operands, the subdirectory must be specified before the parent " 37*da2e3ebdSchin "so that the parent directory will be empty when \brmdir\b attempts " 38*da2e3ebdSchin "to remove it.]" 39*da2e3ebdSchin "[e:ignore-fail-on-non-empty?Ignore each non-empty directory failure.]" 40*da2e3ebdSchin "[p:parents?Remove each explicit \adirectory\a argument directory that " 41*da2e3ebdSchin "becomes empty after its child directories are removed.]" 42*da2e3ebdSchin "[s:suppress?Suppress the message printed on the standard error when " 43*da2e3ebdSchin "\b-p\b is in effect.]" 44*da2e3ebdSchin "\n" 45*da2e3ebdSchin "\ndirectory ...\n" 46*da2e3ebdSchin "\n" 47*da2e3ebdSchin "[+EXIT STATUS?]{" 48*da2e3ebdSchin "[+0?All directories deleted successfully.]" 49*da2e3ebdSchin "[+>0?One or more directories could not be deleted.]" 50*da2e3ebdSchin "}" 51*da2e3ebdSchin "[+SEE ALSO?\bmkdir\b(1), \brm\b(1), \brmdir\b(2), \bunlink\b(2)]" 52*da2e3ebdSchin ; 53*da2e3ebdSchin 54*da2e3ebdSchin #include <cmd.h> 55*da2e3ebdSchin 56*da2e3ebdSchin int 57*da2e3ebdSchin b_rmdir(int argc, char** argv, void* context) 58*da2e3ebdSchin { 59*da2e3ebdSchin register char* dir; 60*da2e3ebdSchin register char* end; 61*da2e3ebdSchin register int n; 62*da2e3ebdSchin int eflag = 0; 63*da2e3ebdSchin int pflag = 0; 64*da2e3ebdSchin int sflag = 0; 65*da2e3ebdSchin 66*da2e3ebdSchin cmdinit(argc, argv, context, ERROR_CATALOG, 0); 67*da2e3ebdSchin while (n = optget(argv, usage)) switch (n) 68*da2e3ebdSchin { 69*da2e3ebdSchin case 'e': 70*da2e3ebdSchin eflag = 1; 71*da2e3ebdSchin break; 72*da2e3ebdSchin case 'p': 73*da2e3ebdSchin pflag = 1; 74*da2e3ebdSchin break; 75*da2e3ebdSchin case 's': 76*da2e3ebdSchin sflag = 1; 77*da2e3ebdSchin break; 78*da2e3ebdSchin case ':': 79*da2e3ebdSchin error(2, "%s", opt_info.arg); 80*da2e3ebdSchin break; 81*da2e3ebdSchin case '?': 82*da2e3ebdSchin error(ERROR_usage(2), "%s", opt_info.arg); 83*da2e3ebdSchin break; 84*da2e3ebdSchin } 85*da2e3ebdSchin argv += opt_info.index; 86*da2e3ebdSchin if (error_info.errors || !*argv) 87*da2e3ebdSchin error(ERROR_usage(2), "%s", optusage(NiL)); 88*da2e3ebdSchin if (!pflag) 89*da2e3ebdSchin sflag = 0; 90*da2e3ebdSchin while (dir = *argv++) 91*da2e3ebdSchin { 92*da2e3ebdSchin end = dir; 93*da2e3ebdSchin if (pflag) end += strlen(dir); 94*da2e3ebdSchin n = 0; 95*da2e3ebdSchin for (;;) 96*da2e3ebdSchin { 97*da2e3ebdSchin if (rmdir(dir) < 0) 98*da2e3ebdSchin { 99*da2e3ebdSchin if (!eflag || errno != EEXIST 100*da2e3ebdSchin #ifdef ENOTEMPTY 101*da2e3ebdSchin && errno != ENOTEMPTY 102*da2e3ebdSchin #endif 103*da2e3ebdSchin ) 104*da2e3ebdSchin { 105*da2e3ebdSchin if (sflag) 106*da2e3ebdSchin error_info.errors++; 107*da2e3ebdSchin else 108*da2e3ebdSchin error(ERROR_system(0), "%s: cannot remove", dir); 109*da2e3ebdSchin } 110*da2e3ebdSchin break; 111*da2e3ebdSchin } 112*da2e3ebdSchin if (n) *end = '/'; 113*da2e3ebdSchin else n = 1; 114*da2e3ebdSchin do if (end <= dir) goto next; while (*--end != '/'); 115*da2e3ebdSchin do if (end <= dir) goto next; while (*(end - 1) == '/' && end--); 116*da2e3ebdSchin *end = 0; 117*da2e3ebdSchin } 118*da2e3ebdSchin next: ; 119*da2e3ebdSchin } 120*da2e3ebdSchin return(error_info.errors != 0); 121*da2e3ebdSchin } 122*da2e3ebdSchin 123