* tmpfile-safer.c: New file.
[gnulib.git] / lib / tmpfile-safer.c
1 /* Invoke tmpfile, but avoid some glitches.
2    Copyright (C) 2006 Free Software Foundation, Inc.
3
4    This program is free software; you can redistribute it and/or modify
5    it under the terms of the GNU General Public License as published by
6    the Free Software Foundation; either version 2, or (at your option)
7    any later version.
8
9    This program is distributed in the hope that it will be useful,
10    but WITHOUT ANY WARRANTY; without even the implied warranty of
11    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12    GNU General Public License for more details.
13
14    You should have received a copy of the GNU General Public License
15    along with this program; if not, write to the Free Software Foundation,
16    Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.  */
17
18 /* Written by Eric Blake, based on ideas from Paul Eggert.  */
19
20 #ifdef HAVE_CONFIG_H
21 # include <config.h>
22 #endif
23
24 #include "stdio-safer.h"
25
26 #include <errno.h>
27 #include <unistd.h>
28 #include "unistd-safer.h"
29
30 #include "binary-io.h"
31
32 #ifndef STDERR_FILENO
33 # define STDERR_FILENO 2
34 #endif
35
36 /* Like tmpfile, but do not return stdin, stdout, or stderr.
37
38    Remember that tmpfile can leave files behind if your program calls _exit,
39    so this function should not be mixed with the close_stdout module.  */
40
41 FILE *
42 tmpfile_safer (void)
43 {
44   FILE *fp = tmpfile ();
45
46   if (fp)
47     {
48       int fd = fileno (fp);
49
50       if (0 <= fd && fd <= STDERR_FILENO)
51         {
52           int f = dup_safer (fd);
53
54           if (f < 0)
55             {
56               int e = errno;
57               fclose (fp);
58               errno = e;
59               return NULL;
60             }
61
62           /* Keep the temporary file in binary mode, on platforms
63              where that matters.  */
64           if (fclose (fp) != 0
65               || ! (fp = fdopen (f, O_BINARY ? "wb+" : "w+")))
66             {
67               int e = errno;
68               close (f);
69               errno = e;
70               return NULL;
71             }
72         }
73     }
74
75   return fp;
76 }