Blame gnulib-tests/sockets.c

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