xref: /titanic_51/usr/src/lib/libcmd/common/mktemp.c (revision 34f9b3eef6fdadbda0a846aa4d68691ac40eace5)
1*34f9b3eeSRoland Mainz /***********************************************************************
2*34f9b3eeSRoland Mainz *                                                                      *
3*34f9b3eeSRoland Mainz *               This software is part of the ast package               *
4*34f9b3eeSRoland Mainz *          Copyright (c) 1992-2009 AT&T Intellectual Property          *
5*34f9b3eeSRoland Mainz *                      and is licensed under the                       *
6*34f9b3eeSRoland Mainz *                  Common Public License, Version 1.0                  *
7*34f9b3eeSRoland Mainz *                    by AT&T Intellectual Property                     *
8*34f9b3eeSRoland Mainz *                                                                      *
9*34f9b3eeSRoland Mainz *                A copy of the License is available at                 *
10*34f9b3eeSRoland Mainz *            http://www.opensource.org/licenses/cpl1.0.txt             *
11*34f9b3eeSRoland Mainz *         (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9)         *
12*34f9b3eeSRoland Mainz *                                                                      *
13*34f9b3eeSRoland Mainz *              Information and Software Systems Research               *
14*34f9b3eeSRoland Mainz *                            AT&T Research                             *
15*34f9b3eeSRoland Mainz *                           Florham Park NJ                            *
16*34f9b3eeSRoland Mainz *                                                                      *
17*34f9b3eeSRoland Mainz *                 Glenn Fowler <gsf@research.att.com>                  *
18*34f9b3eeSRoland Mainz *                  David Korn <dgk@research.att.com>                   *
19*34f9b3eeSRoland Mainz *                                                                      *
20*34f9b3eeSRoland Mainz ***********************************************************************/
21*34f9b3eeSRoland Mainz #pragma prototyped
22*34f9b3eeSRoland Mainz 
23*34f9b3eeSRoland Mainz static const char usage[] =
24*34f9b3eeSRoland Mainz "[-?\n@(#)$Id: mktemp (AT&T Research) 2009-06-19 $\n]"
25*34f9b3eeSRoland Mainz USAGE_LICENSE
26*34f9b3eeSRoland Mainz "[+NAME?mktemp - make temporary file or directory]"
27*34f9b3eeSRoland Mainz "[+DESCRIPTION?\bmktemp\b creates a temporary file with optional base "
28*34f9b3eeSRoland Mainz     "name prefix \aprefix\a. If \aprefix\a is omitted then \btmp_\b is used "
29*34f9b3eeSRoland Mainz     "and \b--tmp\b is implied. If \aprefix\a contains a directory prefix "
30*34f9b3eeSRoland Mainz     "then that directory overrides any of the directories described below. A "
31*34f9b3eeSRoland Mainz     "temporary file will have mode \brw-------\b and a temporary directory "
32*34f9b3eeSRoland Mainz     "will have mode \brwx------\b, subject to \bumask\b(1). Generated paths "
33*34f9b3eeSRoland Mainz     "have these attributes:]"
34*34f9b3eeSRoland Mainz     "{"
35*34f9b3eeSRoland Mainz         "[+*?Lower case to avoid clashes on case ignorant filesystems.]"
36*34f9b3eeSRoland Mainz         "[+*?Pseudo-random part to deter denial of service attacks.]"
37*34f9b3eeSRoland Mainz         "[+*?Pseudo-random part is \a3-chars\a.\a3-chars\a to accomodate "
38*34f9b3eeSRoland Mainz             "8.3 filesystems.]"
39*34f9b3eeSRoland Mainz     "}"
40*34f9b3eeSRoland Mainz "[+?A consecutive sequence of \bX\b's in \aprefix\a is replaced by the "
41*34f9b3eeSRoland Mainz     "pseudo-random part. If there are no \bX\b's then the pseudo-random part "
42*34f9b3eeSRoland Mainz     "is appended to the prefix.]"
43*34f9b3eeSRoland Mainz "[d:directory?Create a directory instead of a regular file.]"
44*34f9b3eeSRoland Mainz "[m:mode]:[mode?Set the mode of the created temporary to \amode\a. "
45*34f9b3eeSRoland Mainz     "\amode\a is symbolic or octal mode as in \bchmod\b(1). Relative modes "
46*34f9b3eeSRoland Mainz     "assume an initial mode of \bu=rwx\b.]"
47*34f9b3eeSRoland Mainz "[p:default?Use \adirectory\a if the \bTMPDIR\b environment variable is "
48*34f9b3eeSRoland Mainz     "not defined. Implies \b--tmp\b.]:[directory]"
49*34f9b3eeSRoland Mainz "[q:quiet?Suppress file and directory error diagnostics.]"
50*34f9b3eeSRoland Mainz "[t:tmp|temporary-directory?Create a path rooted in a temporary "
51*34f9b3eeSRoland Mainz     "directory.]"
52*34f9b3eeSRoland Mainz "[u:unsafe|dry-run?Check for file/directory existence but do not create. "
53*34f9b3eeSRoland Mainz     "Who would want to do that.]"
54*34f9b3eeSRoland Mainz "\n"
55*34f9b3eeSRoland Mainz "\n[ prefix ]\n"
56*34f9b3eeSRoland Mainz "\n"
57*34f9b3eeSRoland Mainz "[+SEE ALSO?\bmkdir\b(1), \bpathtemp\b(3), \bmktemp\b(3)]"
58*34f9b3eeSRoland Mainz ;
59*34f9b3eeSRoland Mainz 
60*34f9b3eeSRoland Mainz #include <cmd.h>
61*34f9b3eeSRoland Mainz #include <ls.h>
62*34f9b3eeSRoland Mainz 
63*34f9b3eeSRoland Mainz int
64*34f9b3eeSRoland Mainz b_mktemp(int argc, char** argv, void* context)
65*34f9b3eeSRoland Mainz {
66*34f9b3eeSRoland Mainz 	mode_t		mode = 0;
67*34f9b3eeSRoland Mainz 	mode_t		mask;
68*34f9b3eeSRoland Mainz 	int		fd;
69*34f9b3eeSRoland Mainz 	int		i;
70*34f9b3eeSRoland Mainz 	int		quiet = 0;
71*34f9b3eeSRoland Mainz 	int		unsafe = 0;
72*34f9b3eeSRoland Mainz 	int*		fdp = &fd;
73*34f9b3eeSRoland Mainz 	char*		dir = "";
74*34f9b3eeSRoland Mainz 	char*		pfx;
75*34f9b3eeSRoland Mainz 	char*		t;
76*34f9b3eeSRoland Mainz 	char		path[PATH_MAX];
77*34f9b3eeSRoland Mainz 
78*34f9b3eeSRoland Mainz 	cmdinit(argc, argv, context, ERROR_CATALOG, ERROR_NOTIFY);
79*34f9b3eeSRoland Mainz 	for (;;)
80*34f9b3eeSRoland Mainz 	{
81*34f9b3eeSRoland Mainz 		switch (optget(argv, usage))
82*34f9b3eeSRoland Mainz 		{
83*34f9b3eeSRoland Mainz 		case 0:
84*34f9b3eeSRoland Mainz 			break;
85*34f9b3eeSRoland Mainz 		case 'd':
86*34f9b3eeSRoland Mainz 			fdp = 0;
87*34f9b3eeSRoland Mainz 			continue;
88*34f9b3eeSRoland Mainz 		case 'm':
89*34f9b3eeSRoland Mainz 			mode = strperm(pfx = opt_info.arg, &opt_info.arg, S_IRWXU);
90*34f9b3eeSRoland Mainz 			if (*opt_info.arg)
91*34f9b3eeSRoland Mainz 				error(ERROR_exit(0), "%s: invalid mode", pfx);
92*34f9b3eeSRoland Mainz 			continue;
93*34f9b3eeSRoland Mainz 		case 'p':
94*34f9b3eeSRoland Mainz 			if ((t = getenv("TMPDIR")) && *t)
95*34f9b3eeSRoland Mainz 				dir = 0;
96*34f9b3eeSRoland Mainz 			else
97*34f9b3eeSRoland Mainz 				dir = opt_info.arg;
98*34f9b3eeSRoland Mainz 			continue;
99*34f9b3eeSRoland Mainz 		case 'q':
100*34f9b3eeSRoland Mainz 			quiet = 1;
101*34f9b3eeSRoland Mainz 			continue;
102*34f9b3eeSRoland Mainz 		case 't':
103*34f9b3eeSRoland Mainz 			dir = 0;
104*34f9b3eeSRoland Mainz 			continue;
105*34f9b3eeSRoland Mainz 		case 'u':
106*34f9b3eeSRoland Mainz 			unsafe = 1;
107*34f9b3eeSRoland Mainz 			fdp = 0;
108*34f9b3eeSRoland Mainz 			continue;
109*34f9b3eeSRoland Mainz 		case ':':
110*34f9b3eeSRoland Mainz 			error(2, "%s", opt_info.arg);
111*34f9b3eeSRoland Mainz 			continue;
112*34f9b3eeSRoland Mainz 		case '?':
113*34f9b3eeSRoland Mainz 			error(ERROR_usage(2), "%s", opt_info.arg);
114*34f9b3eeSRoland Mainz 			continue;
115*34f9b3eeSRoland Mainz 		}
116*34f9b3eeSRoland Mainz 		break;
117*34f9b3eeSRoland Mainz 	}
118*34f9b3eeSRoland Mainz 	argv += opt_info.index;
119*34f9b3eeSRoland Mainz 	if (error_info.errors || (pfx = *argv++) && *argv)
120*34f9b3eeSRoland Mainz 		error(ERROR_usage(2), "%s", optusage(NiL));
121*34f9b3eeSRoland Mainz 	mask = umask(0);
122*34f9b3eeSRoland Mainz 	if (!mode)
123*34f9b3eeSRoland Mainz 		mode = (fdp ? (S_IRUSR|S_IWUSR) : S_IRWXU) & ~mask;
124*34f9b3eeSRoland Mainz 	umask(~mode & (S_IRWXU|S_IRWXG|S_IRWXO));
125*34f9b3eeSRoland Mainz 	if (!pfx)
126*34f9b3eeSRoland Mainz 	{
127*34f9b3eeSRoland Mainz 		pfx = "tmp_";
128*34f9b3eeSRoland Mainz 		if (dir && !*dir)
129*34f9b3eeSRoland Mainz 			dir = 0;
130*34f9b3eeSRoland Mainz 	}
131*34f9b3eeSRoland Mainz 	if (t = strrchr(pfx, '/'))
132*34f9b3eeSRoland Mainz 	{
133*34f9b3eeSRoland Mainz 		i = ++t - pfx;
134*34f9b3eeSRoland Mainz 		dir = fmtbuf(i);
135*34f9b3eeSRoland Mainz 		memcpy(dir, pfx, i);
136*34f9b3eeSRoland Mainz 		dir[i] = 0;
137*34f9b3eeSRoland Mainz 		pfx = t;
138*34f9b3eeSRoland Mainz 	}
139*34f9b3eeSRoland Mainz 	for (;;)
140*34f9b3eeSRoland Mainz 	{
141*34f9b3eeSRoland Mainz 		if (!pathtemp(path, sizeof(path), dir, pfx, fdp))
142*34f9b3eeSRoland Mainz 		{
143*34f9b3eeSRoland Mainz 			if (quiet)
144*34f9b3eeSRoland Mainz 				error_info.errors++;
145*34f9b3eeSRoland Mainz 			else
146*34f9b3eeSRoland Mainz 				error(ERROR_SYSTEM|2, "cannot create temporary path");
147*34f9b3eeSRoland Mainz 			break;
148*34f9b3eeSRoland Mainz 		}
149*34f9b3eeSRoland Mainz 		if (fdp || unsafe || !mkdir(path, mode))
150*34f9b3eeSRoland Mainz 		{
151*34f9b3eeSRoland Mainz 			if (fdp)
152*34f9b3eeSRoland Mainz 				close(*fdp);
153*34f9b3eeSRoland Mainz 			sfputr(sfstdout, path, '\n');
154*34f9b3eeSRoland Mainz 			break;
155*34f9b3eeSRoland Mainz 		}
156*34f9b3eeSRoland Mainz 		if (sh_checksig(context))
157*34f9b3eeSRoland Mainz 		{
158*34f9b3eeSRoland Mainz 			error_info.errors++;
159*34f9b3eeSRoland Mainz 			break;
160*34f9b3eeSRoland Mainz 		}
161*34f9b3eeSRoland Mainz 	}
162*34f9b3eeSRoland Mainz 	umask(mask);
163*34f9b3eeSRoland Mainz 	return error_info.errors != 0;
164*34f9b3eeSRoland Mainz }
165