Blame gnulib-tests/sockets.c

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