1*09d4459fSDaniel Fojt /* Adjust a file descriptor result so that it avoids clobbering
2*09d4459fSDaniel Fojt STD{IN,OUT,ERR}_FILENO, with specific flags.
3*09d4459fSDaniel Fojt
4*09d4459fSDaniel Fojt Copyright (C) 2005-2006, 2009-2020 Free Software Foundation, Inc.
5*09d4459fSDaniel Fojt
6*09d4459fSDaniel Fojt This program is free software: you can redistribute it and/or modify
7*09d4459fSDaniel Fojt it under the terms of the GNU General Public License as published by
8*09d4459fSDaniel Fojt the Free Software Foundation; either version 3 of the License, or
9*09d4459fSDaniel Fojt (at your option) any later version.
10*09d4459fSDaniel Fojt
11*09d4459fSDaniel Fojt This program is distributed in the hope that it will be useful,
12*09d4459fSDaniel Fojt but WITHOUT ANY WARRANTY; without even the implied warranty of
13*09d4459fSDaniel Fojt MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14*09d4459fSDaniel Fojt GNU General Public License for more details.
15*09d4459fSDaniel Fojt
16*09d4459fSDaniel Fojt You should have received a copy of the GNU General Public License
17*09d4459fSDaniel Fojt along with this program. If not, see <https://www.gnu.org/licenses/>. */
18*09d4459fSDaniel Fojt
19*09d4459fSDaniel Fojt /* Written by Paul Eggert and Eric Blake. */
20*09d4459fSDaniel Fojt
21*09d4459fSDaniel Fojt #include <config.h>
22*09d4459fSDaniel Fojt
23*09d4459fSDaniel Fojt /* Specification. */
24*09d4459fSDaniel Fojt #include "unistd-safer.h"
25*09d4459fSDaniel Fojt
26*09d4459fSDaniel Fojt #include <errno.h>
27*09d4459fSDaniel Fojt #include <unistd.h>
28*09d4459fSDaniel Fojt
29*09d4459fSDaniel Fojt /* Return FD, unless FD would be a copy of standard input, output, or
30*09d4459fSDaniel Fojt error; in that case, return a duplicate of FD, closing FD. If FLAG
31*09d4459fSDaniel Fojt contains O_CLOEXEC, the returned FD will have close-on-exec
32*09d4459fSDaniel Fojt semantics. On failure to duplicate, close FD, set errno, and
33*09d4459fSDaniel Fojt return -1. Preserve errno if FD is negative, so that the caller
34*09d4459fSDaniel Fojt can always inspect errno when the returned value is negative.
35*09d4459fSDaniel Fojt
36*09d4459fSDaniel Fojt This function is usefully wrapped around functions that return file
37*09d4459fSDaniel Fojt descriptors, e.g., fd_safer_flag (open ("file", O_RDONLY | flag), flag). */
38*09d4459fSDaniel Fojt
39*09d4459fSDaniel Fojt int
fd_safer_flag(int fd,int flag)40*09d4459fSDaniel Fojt fd_safer_flag (int fd, int flag)
41*09d4459fSDaniel Fojt {
42*09d4459fSDaniel Fojt if (STDIN_FILENO <= fd && fd <= STDERR_FILENO)
43*09d4459fSDaniel Fojt {
44*09d4459fSDaniel Fojt int f = dup_safer_flag (fd, flag);
45*09d4459fSDaniel Fojt int e = errno;
46*09d4459fSDaniel Fojt close (fd);
47*09d4459fSDaniel Fojt errno = e;
48*09d4459fSDaniel Fojt fd = f;
49*09d4459fSDaniel Fojt }
50*09d4459fSDaniel Fojt
51*09d4459fSDaniel Fojt return fd;
52*09d4459fSDaniel Fojt }
53