xref: /titanic_51/usr/src/lib/libcmd/common/rmdir.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  * 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