exp2l: Work around a NetBSD 10.0/i386 bug.
[gnulib.git] / lib / dup3.c
bloba810d3be19ba845de84286c2b70d88f82045427c
1 /* Copy a file descriptor, applying specific flags.
2 Copyright (C) 2009-2024 Free Software Foundation, Inc.
4 This file is free software: you can redistribute it and/or modify
5 it under the terms of the GNU Lesser General Public License as
6 published by the Free Software Foundation, either version 3 of the
7 License, or (at your option) any later version.
9 This file 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 Lesser General Public License for more details.
14 You should have received a copy of the GNU Lesser General Public License
15 along with this program. If not, see <https://www.gnu.org/licenses/>. */
17 #include <config.h>
19 /* Specification. */
20 #include <unistd.h>
22 #include <errno.h>
23 #include <fcntl.h>
24 #include <limits.h>
26 #include "binary-io.h"
28 int
29 dup3 (int oldfd, int newfd, int flags)
31 #if HAVE_DUP3
32 # undef dup3
33 # if HAVE_SETDTABLESIZE
34 /* Avoid a cygwin crasher. */
35 setdtablesize (newfd + 1);
36 # endif
37 /* Try the system call first, if it exists. (We may be running with a glibc
38 that has the function but with an older kernel that lacks it.) */
40 /* Cache the information whether the system call really exists. */
41 static int have_dup3_really; /* 0 = unknown, 1 = yes, -1 = no */
42 if (have_dup3_really >= 0)
44 int result = dup3 (oldfd, newfd, flags);
45 if (!(result < 0 && errno == ENOSYS))
47 have_dup3_really = 1;
48 # if REPLACE_FCHDIR
49 if (0 <= result)
50 result = _gl_register_dup (oldfd, newfd);
51 # endif
52 return result;
54 have_dup3_really = -1;
57 #endif
59 if (newfd < 0 || newfd >= getdtablesize () || fcntl (oldfd, F_GETFD) == -1)
61 errno = EBADF;
62 return -1;
65 if (newfd == oldfd)
67 errno = EINVAL;
68 return -1;
71 /* Check the supported flags.
72 Note that O_NONBLOCK is not supported, because setting it on newfd
73 would implicitly also set it on oldfd. */
74 if ((flags & ~(O_CLOEXEC | O_BINARY | O_TEXT)) != 0)
76 errno = EINVAL;
77 return -1;
80 if (flags & O_CLOEXEC)
82 int result;
83 close (newfd);
84 result = fcntl (oldfd, F_DUPFD_CLOEXEC, newfd);
85 if (newfd < result)
87 close (result);
88 errno = EIO;
89 result = -1;
91 if (result < 0)
92 return -1;
94 else if (dup2 (oldfd, newfd) < 0)
95 return -1;
97 #if O_BINARY
98 if (flags & O_BINARY)
99 set_binary_mode (newfd, O_BINARY);
100 else if (flags & O_TEXT)
101 set_binary_mode (newfd, O_TEXT);
102 #endif
104 return newfd;