2 * tmpfile.c - functions to create and safely open temp files for the shell.
5 /* Copyright (C) 2000 Free Software Foundation, Inc.
7 This file is part of GNU Bash, the Bourne Again SHell.
9 Bash is free software; you can redistribute it and/or modify it
10 under the terms of the GNU General Public License as published by
11 the Free Software Foundation; either version 2, or (at your option)
14 Bash is distributed in the hope that it will be useful, but WITHOUT
15 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
16 or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public
17 License for more details.
19 You should have received a copy of the GNU General Public License
20 along with Bash; see the file COPYING. If not, write to the Free
21 Software Foundation, 59 Temple Place, Suite 330, Boston, MA 02111 USA. */
25 #include <bashtypes.h>
26 #include <posixstat.h>
29 #if defined (HAVE_UNISTD_H)
42 #define BASEOPENFLAGS (O_CREAT | O_TRUNC | O_EXCL)
44 #define DEFAULT_TMPDIR "." /* bogus default, should be changed */
45 #define DEFAULT_NAMEROOT "shtmp"
47 extern pid_t dollar_dollar_pid;
49 static char *sys_tmpdir = (char *)NULL;
51 static int tmpnamelen = -1;
52 static unsigned long filenum = 1L;
63 sys_tmpdir = P_tmpdir;
64 if (stat (sys_tmpdir, &sb) == 0)
69 if (stat (sys_tmpdir, &sb) == 0)
72 sys_tmpdir = "/var/tmp";
73 if (stat (sys_tmpdir, &sb) == 0)
76 sys_tmpdir = "/usr/tmp";
77 if (stat (sys_tmpdir, &sb) == 0)
80 sys_tmpdir = DEFAULT_TMPDIR;
91 tdir = (flags & MT_USETMPDIR) ? get_string_value ("TMPDIR") : (char *)NULL;
93 tdir = get_sys_tmpdir ();
95 #if defined (HAVE_PATHCONF) && defined (_PC_NAME_MAX)
97 tmpnamelen = pathconf (tdir, _PC_NAME_MAX);
106 sh_mktmpname (nameroot, flags)
110 char *filename, *tdir, *lroot;
114 filename = (char *)xmalloc (PATH_MAX + 1);
115 tdir = get_tmpdir (flags);
116 tdlen = strlen (tdir);
118 lroot = nameroot ? nameroot : DEFAULT_NAMEROOT;
121 sprintf (filename, "%s/%s.XXXXXX", tdir, lroot);
122 if (mktemp (filename) == 0)
127 #else /* !USE_MKTEMP */
130 filenum = (filenum << 1) ^
131 (unsigned long) time ((time_t *)0) ^
132 (unsigned long) dollar_dollar_pid ^
133 (unsigned long) ((flags & MT_USERANDOM) ? get_random_number () : ntmpfiles++);
134 sprintf (filename, "%s/%s-%lu", tdir, lroot, filenum);
135 if (tmpnamelen > 0 && tmpnamelen < 32)
136 filename[tdlen + 1 + tmpnamelen] = '\0';
138 r = lstat (filename, &sb);
140 r = stat (filename, &sb);
142 if (r < 0 && errno == ENOENT)
145 #endif /* !USE_MKTEMP */
151 sh_mktmpfd (nameroot, flags, namep)
156 char *filename, *tdir, *lroot;
159 filename = (char *)xmalloc (PATH_MAX + 1);
160 tdir = get_tmpdir (flags);
161 tdlen = strlen (tdir);
163 lroot = nameroot ? nameroot : DEFAULT_NAMEROOT;
166 sprintf (filename, "%s/%s.XXXXXX", tdir, lroot);
167 fd = mkstemp (filename);
168 if (fd < 0 || namep == 0)
176 #else /* !USE_MKSTEMP */
179 filenum = (filenum << 1) ^
180 (unsigned long) time ((time_t *)0) ^
181 (unsigned long) dollar_dollar_pid ^
182 (unsigned long) ((flags & MT_USERANDOM) ? get_random_number () : ntmpfiles++);
183 sprintf (filename, "%s/%s-%lu", tdir, lroot, filenum);
184 if (tmpnamelen > 0 && tmpnamelen < 32)
185 filename[tdlen + 1 + tmpnamelen] = '\0';
186 fd = open (filename, BASEOPENFLAGS | ((flags & MT_READWRITE) ? O_RDWR : O_WRONLY), 0600);
188 while (fd < 0 && errno == EEXIST);
196 #endif /* !USE_MKSTEMP */
200 sh_mktmpfp (nameroot, flags, namep)
208 fd = sh_mktmpfd (nameroot, flags, namep);
210 return ((FILE *)NULL);
211 fp = fdopen (fd, (flags & MT_READWRITE) ? "w+" : "w");