mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2024-12-27 04:52:05 +08:00
6ec2e0f5bd
These two modules are necessary because of the rework that will be done in the "change directory" logic on GDB/gdbserver in the next commits. First, we will get rid of the "gdb_dirbuf" global variable and instead rely on the fact that "getcwd (NULL, 0)", which is a GNU extension, returns a heap-allocated string with the necessary bytes to hold the full path. This is a good practice not only because globals are not ideal but also because there is no good way to know beforehand the size of the full pathname allowed in the filesystem ("PATH_MAX" is not portable and does not reflect all the possible filesystems out there). We will also have a way to "cd" to a directory also on gdbserver, but in order to do that uniformly, there must be a way to do tilde expansion on directories provided by the user. Currently, GDB uses "tilde_expand" from readline to do that, but gdbserver doesn't link against readline and therefore cannot use this function. The solution is to use "glob" instead, which can perform tilde expansion as a GNU extension. Therefore, we need gnulib's version of "glob". A special note is necessary for gdb/ser-tcp.c. It defines "close" as "closesocket" on Win32 targets. If we leave the code as is, this would conflict with gnulib's definition of "close". Therefore, in order to keep the same code path from before this import, I decided to "#undef close" first, and then let the original (re)definition of it take place. gdb/ChangeLog: 2017-09-22 Sergio Durigan Junior <sergiodj@redhat.com> * gnulib/aclocal.m4: Regenerate. * gnulib/config.in: Regenerate. * gnulib/configure: Regenerate. * gnulib/import/Makefile.am: Regenerate. * gnulib/import/Makefile.in: Regenerate. * gnulib/import/assure.h: New file. * gnulib/import/at-func.c: Likewise * gnulib/import/chdir-long.c: New file. * gnulib/import/chdir-long.h: New file. * gnulib/import/cloexec.c: New file. * gnulib/import/cloexec.h: New file. * gnulib/import/close.c: New file. * gnulib/import/closedir.c: New file. * gnulib/import/dirent-private.h: New file. * gnulib/import/dup-safer.c: New file. * gnulib/import/dup.c: New file. * gnulib/import/dup2.c: New file. * gnulib/import/error.c: New file. * gnulib/import/error.h: New file. * gnulib/import/exitfail.c: New file. * gnulib/import/exitfail.h: New file. * gnulib/import/fchdir.c: New file. * gnulib/import/fcntl.c: New file. * gnulib/import/fcntl.in.h: New file. * gnulib/import/fd-hook.c: New file. * gnulib/import/fd-hook.h: New file. * gnulib/import/fd-safer.c: New file. * gnulib/import/fdopendir.c: New file. * gnulib/import/filename.h: New file. * gnulib/import/filenamecat-lgpl.c: New file. * gnulib/import/filenamecat.h: New file. * gnulib/import/fstat.c: New file. * gnulib/import/fstatat.c: New file. * gnulib/import/getcwd-lgpl.c: New file. * gnulib/import/getcwd.c: New file. * gnulib/import/getdtablesize.c: New file. * gnulib/import/getlogin_r.c: New file. * gnulib/import/getprogname.c: New file. * gnulib/import/getprogname.h: New file. * gnulib/import/gettext.h: New file. * gnulib/import/glob-libc.h: New file. * gnulib/import/glob.c: New file. * gnulib/import/glob.in.h: New file. * gnulib/import/intprops.h: New file. * gnulib/import/m4/chdir-long.m4: New file. * gnulib/import/m4/close.m4: New file. * gnulib/import/m4/closedir.m4: New file. * gnulib/import/m4/d-ino.m4: New file. * gnulib/import/m4/d-type.m4: New file. * gnulib/import/m4/dup.m4: New file. * gnulib/import/m4/dup2.m4: New file. * gnulib/import/m4/error.m4: New file. * gnulib/import/m4/fchdir.m4: New file. * gnulib/import/m4/fcntl.m4: New file. * gnulib/import/m4/fcntl_h.m4: New file. * gnulib/import/m4/fdopendir.m4: New file. * gnulib/import/m4/filenamecat.m4: New file. * gnulib/import/m4/fstat.m4: New file. * gnulib/import/m4/fstatat.m4: New file. * gnulib/import/m4/getcwd-abort-bug.m4: New file. * gnulib/import/m4/getcwd-path-max.m4: New file. * gnulib/import/m4/getcwd.m4: New file. * gnulib/import/m4/getdtablesize.m4: New file. * gnulib/import/m4/getlogin_r.m4: New file. * gnulib/import/m4/getprogname.m4: New file. * gnulib/import/m4/glob.m4: New file. * gnulib/import/m4/gnulib-cache.m4: Regenerate. * gnulib/import/m4/gnulib-comp.m4: Regenerate. * gnulib/import/m4/mempcpy.m4: New file. * gnulib/import/m4/memrchr.m4: New file. * gnulib/import/m4/mode_t.m4: New file. * gnulib/import/m4/msvc-inval.m4: New file. * gnulib/import/m4/msvc-nothrow.m4: New file. * gnulib/import/m4/open.m4: New file. * gnulib/import/m4/openat.m4: New file. * gnulib/import/m4/opendir.m4: New file. * gnulib/import/m4/readdir.m4: New file. * gnulib/import/m4/realloc.m4: New file. * gnulib/import/m4/rewinddir.m4: New file. * gnulib/import/m4/save-cwd.m4: New file. * gnulib/import/m4/strdup.m4: New file. * gnulib/import/m4/strerror.m4: New file. * gnulib/import/m4/unistd-safer.m4: New file. * gnulib/import/mempcpy.c: New file. * gnulib/import/memrchr.c: New file. * gnulib/import/msvc-inval.c: New file. * gnulib/import/msvc-inval.h: New file. * gnulib/import/msvc-nothrow.c: New file. * gnulib/import/msvc-nothrow.h: New file. * gnulib/import/open.c: New file. * gnulib/import/openat-die.c: New file. * gnulib/import/openat-priv.h: New file. * gnulib/import/openat-proc.c: New file. * gnulib/import/openat.c: New file. * gnulib/import/openat.h: New file. * gnulib/import/opendir.c: New file. * gnulib/import/pipe-safer.c: New file. * gnulib/import/readdir.c: New file. * gnulib/import/realloc.c: New file. * gnulib/import/rewinddir.c: New file. * gnulib/import/save-cwd.c: New file. * gnulib/import/save-cwd.h: New file. * gnulib/import/strdup.c: New file. * gnulib/import/strerror-override.c: New file. * gnulib/import/strerror-override.h: New file. * gnulib/import/strerror.c: New file. * gnulib/import/unistd--.h: New file. * gnulib/import/unistd-safer.h: New file. * gnulib/update-gnulib.sh (IMPORTED_GNULIB_MODULES): Add "getcwd" and "glob". * ser-tcp.c: Undefine "close" before redefining it.
216 lines
4.6 KiB
C
216 lines
4.6 KiB
C
/* Duplicate an open file descriptor to a specified file descriptor.
|
|
|
|
Copyright (C) 1999, 2004-2007, 2009-2016 Free Software Foundation, Inc.
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
|
|
|
/* written by Paul Eggert */
|
|
|
|
#include <config.h>
|
|
|
|
/* Specification. */
|
|
#include <unistd.h>
|
|
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
|
|
#if HAVE_DUP2
|
|
|
|
# undef dup2
|
|
|
|
# if (defined _WIN32 || defined __WIN32__) && ! defined __CYGWIN__
|
|
|
|
/* Get declarations of the native Windows API functions. */
|
|
# define WIN32_LEAN_AND_MEAN
|
|
# include <windows.h>
|
|
|
|
# include "msvc-inval.h"
|
|
|
|
/* Get _get_osfhandle. */
|
|
# include "msvc-nothrow.h"
|
|
|
|
static int
|
|
ms_windows_dup2 (int fd, int desired_fd)
|
|
{
|
|
int result;
|
|
|
|
/* If fd is closed, mingw hangs on dup2 (fd, fd). If fd is open,
|
|
dup2 (fd, fd) returns 0, but all further attempts to use fd in
|
|
future dup2 calls will hang. */
|
|
if (fd == desired_fd)
|
|
{
|
|
if ((HANDLE) _get_osfhandle (fd) == INVALID_HANDLE_VALUE)
|
|
{
|
|
errno = EBADF;
|
|
return -1;
|
|
}
|
|
return fd;
|
|
}
|
|
|
|
/* Wine 1.0.1 return 0 when desired_fd is negative but not -1:
|
|
http://bugs.winehq.org/show_bug.cgi?id=21289 */
|
|
if (desired_fd < 0)
|
|
{
|
|
errno = EBADF;
|
|
return -1;
|
|
}
|
|
|
|
TRY_MSVC_INVAL
|
|
{
|
|
result = dup2 (fd, desired_fd);
|
|
}
|
|
CATCH_MSVC_INVAL
|
|
{
|
|
errno = EBADF;
|
|
result = -1;
|
|
}
|
|
DONE_MSVC_INVAL;
|
|
|
|
if (result == 0)
|
|
result = desired_fd;
|
|
|
|
return result;
|
|
}
|
|
|
|
# define dup2 ms_windows_dup2
|
|
|
|
# elif defined __KLIBC__
|
|
|
|
# include <InnoTekLIBC/backend.h>
|
|
|
|
static int
|
|
klibc_dup2dirfd (int fd, int desired_fd)
|
|
{
|
|
int tempfd;
|
|
int dupfd;
|
|
|
|
tempfd = open ("NUL", O_RDONLY);
|
|
if (tempfd == -1)
|
|
return -1;
|
|
|
|
if (tempfd == desired_fd)
|
|
{
|
|
close (tempfd);
|
|
|
|
char path[_MAX_PATH];
|
|
if (__libc_Back_ioFHToPath (fd, path, sizeof (path)))
|
|
return -1;
|
|
|
|
return open(path, O_RDONLY);
|
|
}
|
|
|
|
dupfd = klibc_dup2dirfd (fd, desired_fd);
|
|
|
|
close (tempfd);
|
|
|
|
return dupfd;
|
|
}
|
|
|
|
static int
|
|
klibc_dup2 (int fd, int desired_fd)
|
|
{
|
|
int dupfd;
|
|
struct stat sbuf;
|
|
|
|
dupfd = dup2 (fd, desired_fd);
|
|
if (dupfd == -1 && errno == ENOTSUP \
|
|
&& !fstat (fd, &sbuf) && S_ISDIR (sbuf.st_mode))
|
|
{
|
|
close (desired_fd);
|
|
|
|
return klibc_dup2dirfd (fd, desired_fd);
|
|
}
|
|
|
|
return dupfd;
|
|
}
|
|
|
|
# define dup2 klibc_dup2
|
|
# endif
|
|
|
|
int
|
|
rpl_dup2 (int fd, int desired_fd)
|
|
{
|
|
int result;
|
|
|
|
# ifdef F_GETFL
|
|
/* On Linux kernels 2.6.26-2.6.29, dup2 (fd, fd) returns -EBADF.
|
|
On Cygwin 1.5.x, dup2 (1, 1) returns 0.
|
|
On Cygwin 1.7.17, dup2 (1, -1) dumps core.
|
|
On Cygwin 1.7.25, dup2 (1, 256) can dump core.
|
|
On Haiku, dup2 (fd, fd) mistakenly clears FD_CLOEXEC. */
|
|
# if HAVE_SETDTABLESIZE
|
|
setdtablesize (desired_fd + 1);
|
|
# endif
|
|
if (desired_fd < 0)
|
|
fd = desired_fd;
|
|
if (fd == desired_fd)
|
|
return fcntl (fd, F_GETFL) == -1 ? -1 : fd;
|
|
# endif
|
|
|
|
result = dup2 (fd, desired_fd);
|
|
|
|
/* Correct an errno value on FreeBSD 6.1 and Cygwin 1.5.x. */
|
|
if (result == -1 && errno == EMFILE)
|
|
errno = EBADF;
|
|
# if REPLACE_FCHDIR
|
|
if (fd != desired_fd && result != -1)
|
|
result = _gl_register_dup (fd, result);
|
|
# endif
|
|
return result;
|
|
}
|
|
|
|
#else /* !HAVE_DUP2 */
|
|
|
|
/* On older platforms, dup2 did not exist. */
|
|
|
|
# ifndef F_DUPFD
|
|
static int
|
|
dupfd (int fd, int desired_fd)
|
|
{
|
|
int duplicated_fd = dup (fd);
|
|
if (duplicated_fd < 0 || duplicated_fd == desired_fd)
|
|
return duplicated_fd;
|
|
else
|
|
{
|
|
int r = dupfd (fd, desired_fd);
|
|
int e = errno;
|
|
close (duplicated_fd);
|
|
errno = e;
|
|
return r;
|
|
}
|
|
}
|
|
# endif
|
|
|
|
int
|
|
dup2 (int fd, int desired_fd)
|
|
{
|
|
int result = fcntl (fd, F_GETFL) < 0 ? -1 : fd;
|
|
if (result == -1 || fd == desired_fd)
|
|
return result;
|
|
close (desired_fd);
|
|
# ifdef F_DUPFD
|
|
result = fcntl (fd, F_DUPFD, desired_fd);
|
|
# if REPLACE_FCHDIR
|
|
if (0 <= result)
|
|
result = _gl_register_dup (fd, result);
|
|
# endif
|
|
# else
|
|
result = dupfd (fd, desired_fd);
|
|
# endif
|
|
if (result == -1 && (errno == EMFILE || errno == EINVAL))
|
|
errno = EBADF;
|
|
return result;
|
|
}
|
|
#endif /* !HAVE_DUP2 */
|