Blame lib/sockets.c

Packit Service a2489d
/* sockets.c --- wrappers for Windows socket functions
Packit Service a2489d
Packit Service a2489d
   Copyright (C) 2008-2018 Free Software Foundation, Inc.
Packit Service a2489d
Packit Service a2489d
   This program is free software: you can redistribute it and/or modify
Packit Service a2489d
   it under the terms of the GNU General Public License as published by
Packit Service a2489d
   the Free Software Foundation; either version 3 of the License, or
Packit Service a2489d
   (at your option) any later version.
Packit Service a2489d
Packit Service a2489d
   This program is distributed in the hope that it will be useful,
Packit Service a2489d
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service a2489d
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service a2489d
   GNU General Public License for more details.
Packit Service a2489d
Packit Service a2489d
   You should have received a copy of the GNU General Public License
Packit Service a2489d
   along with this program.  If not, see <https://www.gnu.org/licenses/>.  */
Packit Service a2489d
Packit Service a2489d
/* Written by Simon Josefsson */
Packit Service a2489d
Packit Service a2489d
#include <config.h>
Packit Service a2489d
Packit Service a2489d
/* Specification.  */
Packit Service a2489d
#include "sockets.h"
Packit Service a2489d
Packit Service a2489d
#if WINDOWS_SOCKETS
Packit Service a2489d
Packit Service a2489d
/* This includes winsock2.h on MinGW. */
Packit Service a2489d
# include <sys/socket.h>
Packit Service a2489d
Packit Service a2489d
# include "fd-hook.h"
Packit Service a2489d
# if GNULIB_MSVC_NOTHROW
Packit Service a2489d
#  include "msvc-nothrow.h"
Packit Service a2489d
# else
Packit Service a2489d
#  include <io.h>
Packit Service a2489d
# endif
Packit Service a2489d
Packit Service a2489d
/* Get set_winsock_errno, FD_TO_SOCKET etc. */
Packit Service a2489d
# include "w32sock.h"
Packit Service a2489d
Packit Service a2489d
static int
Packit Service a2489d
close_fd_maybe_socket (const struct fd_hook *remaining_list,
Packit Service a2489d
                       gl_close_fn primary,
Packit Service a2489d
                       int fd)
Packit Service a2489d
{
Packit Service a2489d
  /* Note about multithread-safety: There is a race condition where, between
Packit Service a2489d
     our calls to closesocket() and the primary close(), some other thread
Packit Service a2489d
     could make system calls that allocate precisely the same HANDLE value
Packit Service a2489d
     as sock; then the primary close() would call CloseHandle() on it.  */
Packit Service a2489d
  SOCKET sock;
Packit Service a2489d
  WSANETWORKEVENTS ev;
Packit Service a2489d
Packit Service a2489d
  /* Test whether fd refers to a socket.  */
Packit Service a2489d
  sock = FD_TO_SOCKET (fd);
Packit Service a2489d
  ev.lNetworkEvents = 0xDEADBEEF;
Packit Service a2489d
  WSAEnumNetworkEvents (sock, NULL, &ev;;
Packit Service a2489d
  if (ev.lNetworkEvents != 0xDEADBEEF)
Packit Service a2489d
    {
Packit Service a2489d
      /* fd refers to a socket.  */
Packit Service a2489d
      /* FIXME: other applications, like squid, use an undocumented
Packit Service a2489d
         _free_osfhnd free function.  But this is not enough: The 'osfile'
Packit Service a2489d
         flags for fd also needs to be cleared, but it is hard to access it.
Packit Service a2489d
         Instead, here we just close twice the file descriptor.  */
Packit Service a2489d
      if (closesocket (sock))
Packit Service a2489d
        {
Packit Service a2489d
          set_winsock_errno ();
Packit Service a2489d
          return -1;
Packit Service a2489d
        }
Packit Service a2489d
      else
Packit Service a2489d
        {
Packit Service a2489d
          /* This call frees the file descriptor and does a
Packit Service a2489d
             CloseHandle ((HANDLE) _get_osfhandle (fd)), which fails.  */
Packit Service a2489d
          _close (fd);
Packit Service a2489d
          return 0;
Packit Service a2489d
        }
Packit Service a2489d
    }
Packit Service a2489d
  else
Packit Service a2489d
    /* Some other type of file descriptor.  */
Packit Service a2489d
    return execute_close_hooks (remaining_list, primary, fd);
Packit Service a2489d
}
Packit Service a2489d
Packit Service a2489d
static int
Packit Service a2489d
ioctl_fd_maybe_socket (const struct fd_hook *remaining_list,
Packit Service a2489d
                       gl_ioctl_fn primary,
Packit Service a2489d
                       int fd, int request, void *arg)
Packit Service a2489d
{
Packit Service a2489d
  SOCKET sock;
Packit Service a2489d
  WSANETWORKEVENTS ev;
Packit Service a2489d
Packit Service a2489d
  /* Test whether fd refers to a socket.  */
Packit Service a2489d
  sock = FD_TO_SOCKET (fd);
Packit Service a2489d
  ev.lNetworkEvents = 0xDEADBEEF;
Packit Service a2489d
  WSAEnumNetworkEvents (sock, NULL, &ev;;
Packit Service a2489d
  if (ev.lNetworkEvents != 0xDEADBEEF)
Packit Service a2489d
    {
Packit Service a2489d
      /* fd refers to a socket.  */
Packit Service a2489d
      if (ioctlsocket (sock, request, arg) < 0)
Packit Service a2489d
        {
Packit Service a2489d
          set_winsock_errno ();
Packit Service a2489d
          return -1;
Packit Service a2489d
        }
Packit Service a2489d
      else
Packit Service a2489d
        return 0;
Packit Service a2489d
    }
Packit Service a2489d
  else
Packit Service a2489d
    /* Some other type of file descriptor.  */
Packit Service a2489d
    return execute_ioctl_hooks (remaining_list, primary, fd, request, arg);
Packit Service a2489d
}
Packit Service a2489d
Packit Service a2489d
static struct fd_hook fd_sockets_hook;
Packit Service a2489d
Packit Service a2489d
static int initialized_sockets_version /* = 0 */;
Packit Service a2489d
Packit Service a2489d
#endif /* WINDOWS_SOCKETS */
Packit Service a2489d
Packit Service a2489d
int
Packit Service a2489d
gl_sockets_startup (int version _GL_UNUSED)
Packit Service a2489d
{
Packit Service a2489d
#if WINDOWS_SOCKETS
Packit Service a2489d
  if (version > initialized_sockets_version)
Packit Service a2489d
    {
Packit Service a2489d
      WSADATA data;
Packit Service a2489d
      int err;
Packit Service a2489d
Packit Service a2489d
      err = WSAStartup (version, &data);
Packit Service a2489d
      if (err != 0)
Packit Service a2489d
        return 1;
Packit Service a2489d
Packit Service a2489d
      if (data.wVersion != version)
Packit Service a2489d
        {
Packit Service a2489d
          WSACleanup ();
Packit Service a2489d
          return 2;
Packit Service a2489d
        }
Packit Service a2489d
Packit Service a2489d
      if (initialized_sockets_version == 0)
Packit Service a2489d
        register_fd_hook (close_fd_maybe_socket, ioctl_fd_maybe_socket,
Packit Service a2489d
                          &fd_sockets_hook);
Packit Service a2489d
Packit Service a2489d
      initialized_sockets_version = version;
Packit Service a2489d
    }
Packit Service a2489d
#endif
Packit Service a2489d
Packit Service a2489d
  return 0;
Packit Service a2489d
}
Packit Service a2489d
Packit Service a2489d
int
Packit Service a2489d
gl_sockets_cleanup (void)
Packit Service a2489d
{
Packit Service a2489d
#if WINDOWS_SOCKETS
Packit Service a2489d
  int err;
Packit Service a2489d
Packit Service a2489d
  initialized_sockets_version = 0;
Packit Service a2489d
Packit Service a2489d
  unregister_fd_hook (&fd_sockets_hook);
Packit Service a2489d
Packit Service a2489d
  err = WSACleanup ();
Packit Service a2489d
  if (err != 0)
Packit Service a2489d
    return 1;
Packit Service a2489d
#endif
Packit Service a2489d
Packit Service a2489d
  return 0;
Packit Service a2489d
}