xref: /titanic_50/usr/src/lib/libcmd/common/dirname.c (revision 3e14f97f673e8a630f076077de35afdd43dc1587)
1da2e3ebdSchin /***********************************************************************
2da2e3ebdSchin *                                                                      *
3da2e3ebdSchin *               This software is part of the ast package               *
4*3e14f97fSRoger A. Faulkner *          Copyright (c) 1992-2010 AT&T Intellectual Property          *
5da2e3ebdSchin *                      and is licensed under the                       *
6da2e3ebdSchin *                  Common Public License, Version 1.0                  *
77c2fbfb3SApril Chin *                    by AT&T Intellectual Property                     *
8da2e3ebdSchin *                                                                      *
9da2e3ebdSchin *                A copy of the License is available at                 *
10da2e3ebdSchin *            http://www.opensource.org/licenses/cpl1.0.txt             *
11da2e3ebdSchin *         (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9)         *
12da2e3ebdSchin *                                                                      *
13da2e3ebdSchin *              Information and Software Systems Research               *
14da2e3ebdSchin *                            AT&T Research                             *
15da2e3ebdSchin *                           Florham Park NJ                            *
16da2e3ebdSchin *                                                                      *
17da2e3ebdSchin *                 Glenn Fowler <gsf@research.att.com>                  *
18da2e3ebdSchin *                  David Korn <dgk@research.att.com>                   *
19da2e3ebdSchin *                                                                      *
20da2e3ebdSchin ***********************************************************************/
21da2e3ebdSchin #pragma prototyped
22da2e3ebdSchin /*
23da2e3ebdSchin  * David Korn
24da2e3ebdSchin  * AT&T Bell Laboratories
25da2e3ebdSchin  *
26da2e3ebdSchin  * dirname path [suffix]
27da2e3ebdSchin  *
28da2e3ebdSchin  * print the dirname of a pathname
29da2e3ebdSchin  */
30da2e3ebdSchin 
31da2e3ebdSchin static const char usage[] =
3234f9b3eeSRoland Mainz "[-?\n@(#)$Id: dirname (AT&T Research) 2009-01-31 $\n]"
33da2e3ebdSchin USAGE_LICENSE
34da2e3ebdSchin "[+NAME?dirname - return directory portion of file name]"
35da2e3ebdSchin "[+DESCRIPTION?\bdirname\b treats \astring\a as a file name and returns "
36da2e3ebdSchin 	"the name of the directory containing the file name by deleting "
37da2e3ebdSchin 	"the last component from \astring\a.]"
38da2e3ebdSchin "[+?If \astring\a consists solely of \b/\b characters the output will "
39da2e3ebdSchin 	"be a single \b/\b unless \bPATH_LEADING_SLASHES\b returned by "
40da2e3ebdSchin 	"\bgetconf\b(1) is \b1\b and \astring\a consists of multiple "
41da2e3ebdSchin 	"\b/\b characters in which case \b//\b will be output.  "
42da2e3ebdSchin 	"Otherwise, trailing \b/\b characters are removed, and if "
43da2e3ebdSchin 	"there are no remaining \b/\b characters in \astring\a, "
44da2e3ebdSchin 	"the string \b.\b will be written to standard output.  "
45da2e3ebdSchin 	"Otherwise, all characters following the last \b/\b are removed. "
46da2e3ebdSchin 	"If the remaining string consists solely of \b/\b characters, "
47da2e3ebdSchin 	"the output will be as if the original string had consisted solely "
48da2e3ebdSchin 	"as \b/\b characters as described above.  Otherwise, all "
49da2e3ebdSchin 	"trailing slashes are removed and the output will be this string "
50da2e3ebdSchin 	"unless this string is empty.  If empty the output will be \b.\b.]"
5134f9b3eeSRoland Mainz "[f:file?Print the \b$PATH\b relative regular file path for \astring\a.]"
5234f9b3eeSRoland Mainz "[r:relative?Print the \b$PATH\b relative readable file path for \astring\a.]"
5334f9b3eeSRoland Mainz "[x:executable?Print the \b$PATH\b relative executable file path for \astring\a.]"
54da2e3ebdSchin "\n"
55da2e3ebdSchin "\nstring\n"
56da2e3ebdSchin "\n"
57da2e3ebdSchin "[+EXIT STATUS?]{"
58da2e3ebdSchin         "[+0?Successful Completion.]"
59da2e3ebdSchin         "[+>0?An error occurred.]"
60da2e3ebdSchin "}"
6134f9b3eeSRoland Mainz "[+SEE ALSO?\bbasename\b(1), \bgetconf\b(1), \bdirname\b(3), \bpathname\b(3)]"
62da2e3ebdSchin ;
63da2e3ebdSchin 
64da2e3ebdSchin #include <cmd.h>
65da2e3ebdSchin 
l_dirname(register Sfio_t * outfile,register const char * pathname)66da2e3ebdSchin static void l_dirname(register Sfio_t *outfile, register const char *pathname)
67da2e3ebdSchin {
68da2e3ebdSchin 	register const char  *last;
69da2e3ebdSchin 	/* go to end of path */
70da2e3ebdSchin 	for(last=pathname; *last; last++);
71da2e3ebdSchin 	/* back over trailing '/' */
72da2e3ebdSchin 	while(last>pathname && *--last=='/');
73da2e3ebdSchin 	/* back over non-slash chars */
74da2e3ebdSchin 	for(;last>pathname && *last!='/';last--);
75da2e3ebdSchin 	if(last==pathname)
76da2e3ebdSchin 	{
77da2e3ebdSchin 		/* all '/' or "" */
78da2e3ebdSchin 		if(*pathname!='/')
79da2e3ebdSchin 			last = pathname = ".";
80da2e3ebdSchin 	}
81da2e3ebdSchin 	else
82da2e3ebdSchin 	{
83da2e3ebdSchin 		/* back over trailing '/' */
84da2e3ebdSchin 		for(;*last=='/' && last > pathname; last--);
85da2e3ebdSchin 	}
86da2e3ebdSchin 	/* preserve // */
87da2e3ebdSchin 	if(last!=pathname && pathname[0]=='/' && pathname[1]=='/')
88da2e3ebdSchin 	{
89da2e3ebdSchin 		while(pathname[2]=='/' && pathname<last)
90da2e3ebdSchin 			pathname++;
91da2e3ebdSchin 		if(last!=pathname && pathname[0]=='/' && pathname[1]=='/' && *astconf("PATH_LEADING_SLASHES",NiL,NiL)!='1')
92da2e3ebdSchin 			pathname++;
93da2e3ebdSchin 	}
94da2e3ebdSchin 	sfwrite(outfile,pathname,last+1-pathname);
95da2e3ebdSchin 	sfputc(outfile,'\n');
96da2e3ebdSchin }
97da2e3ebdSchin 
98da2e3ebdSchin int
b_dirname(int argc,register char * argv[],void * context)99da2e3ebdSchin b_dirname(int argc,register char *argv[], void* context)
100da2e3ebdSchin {
101da2e3ebdSchin 	register int n;
10234f9b3eeSRoland Mainz 	int mode = 0;
10334f9b3eeSRoland Mainz 	char buf[PATH_MAX];
104da2e3ebdSchin 
105da2e3ebdSchin 	cmdinit(argc, argv, context, ERROR_CATALOG, 0);
106da2e3ebdSchin 	while (n = optget(argv, usage)) switch (n)
107da2e3ebdSchin 	{
10834f9b3eeSRoland Mainz 	case 'f':
10934f9b3eeSRoland Mainz 		mode |= PATH_REGULAR;
11034f9b3eeSRoland Mainz 		break;
11134f9b3eeSRoland Mainz 	case 'r':
11234f9b3eeSRoland Mainz 		mode &= ~PATH_REGULAR;
11334f9b3eeSRoland Mainz 		mode |= PATH_READ;
11434f9b3eeSRoland Mainz 		break;
11534f9b3eeSRoland Mainz 	case 'x':
11634f9b3eeSRoland Mainz 		mode |= PATH_EXECUTE;
11734f9b3eeSRoland Mainz 		break;
118da2e3ebdSchin 	case ':':
119da2e3ebdSchin 		error(2, "%s", opt_info.arg);
120da2e3ebdSchin 		break;
121da2e3ebdSchin 	case '?':
122da2e3ebdSchin 		error(ERROR_usage(2), "%s", opt_info.arg);
123da2e3ebdSchin 		break;
124da2e3ebdSchin 	}
125da2e3ebdSchin 	argv += opt_info.index;
126da2e3ebdSchin 	argc -= opt_info.index;
127da2e3ebdSchin 	if(error_info.errors || argc != 1)
128da2e3ebdSchin 		error(ERROR_usage(2),"%s", optusage(NiL));
12934f9b3eeSRoland Mainz 	if(!mode)
130da2e3ebdSchin 		l_dirname(sfstdout,argv[0]);
13134f9b3eeSRoland Mainz 	else if(pathpath(buf, argv[0], "", mode))
13234f9b3eeSRoland Mainz 		sfputr(sfstdout, buf, '\n');
13334f9b3eeSRoland Mainz 	else
13434f9b3eeSRoland Mainz 		error(1|ERROR_WARNING, "%s: relative path not found", argv[0]);
135da2e3ebdSchin 	return(0);
136da2e3ebdSchin }
137