Blame gl/fd-hook.c

Packit Service 4684c1
/* Hook for making file descriptor functions close(), ioctl() extensible.
Packit Service 4684c1
   Copyright (C) 2009-2020 Free Software Foundation, Inc.
Packit Service 4684c1
   Written by Bruno Haible <bruno@clisp.org>, 2009.
Packit Service 4684c1
Packit Service 4684c1
   This program is free software: you can redistribute it and/or modify it
Packit Service 4684c1
   under the terms of the GNU Lesser General Public License as published
Packit Service 4684c1
   by the Free Software Foundation; either version 2.1 of the License, or
Packit Service 4684c1
   (at your option) any later version.
Packit Service 4684c1
Packit Service 4684c1
   This program is distributed in the hope that it will be useful,
Packit Service 4684c1
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 4684c1
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 4684c1
   Lesser General Public License for more details.
Packit Service 4684c1
Packit Service 4684c1
   You should have received a copy of the GNU Lesser General Public License
Packit Service 4684c1
   along with this program.  If not, see <https://www.gnu.org/licenses/>.  */
Packit Service 4684c1
Packit Service 4684c1
#include <config.h>
Packit Service 4684c1
Packit Service 4684c1
/* Specification.  */
Packit Service 4684c1
#include "fd-hook.h"
Packit Service 4684c1
Packit Service 4684c1
#include <stdlib.h>
Packit Service 4684c1
Packit Service 4684c1
/* Currently, this entire code is only needed for the handling of sockets
Packit Service 4684c1
   on native Windows platforms.  */
Packit Service 4684c1
#if WINDOWS_SOCKETS
Packit Service 4684c1
Packit Service 4684c1
/* The first and last link in the doubly linked list.
Packit Service 4684c1
   Initially the list is empty.  */
Packit Service 4684c1
static struct fd_hook anchor = { &anchor, &anchor, NULL, NULL };
Packit Service 4684c1
Packit Service 4684c1
int
Packit Service 4684c1
execute_close_hooks (const struct fd_hook *remaining_list, gl_close_fn primary,
Packit Service 4684c1
                     int fd)
Packit Service 4684c1
{
Packit Service 4684c1
  if (remaining_list == &anchor)
Packit Service 4684c1
    /* End of list reached.  */
Packit Service 4684c1
    return primary (fd);
Packit Service 4684c1
  else
Packit Service 4684c1
    return remaining_list->private_close_fn (remaining_list->private_next,
Packit Service 4684c1
                                             primary, fd);
Packit Service 4684c1
}
Packit Service 4684c1
Packit Service 4684c1
int
Packit Service 4684c1
execute_all_close_hooks (gl_close_fn primary, int fd)
Packit Service 4684c1
{
Packit Service 4684c1
  return execute_close_hooks (anchor.private_next, primary, fd);
Packit Service 4684c1
}
Packit Service 4684c1
Packit Service 4684c1
int
Packit Service 4684c1
execute_ioctl_hooks (const struct fd_hook *remaining_list, gl_ioctl_fn primary,
Packit Service 4684c1
                     int fd, int request, void *arg)
Packit Service 4684c1
{
Packit Service 4684c1
  if (remaining_list == &anchor)
Packit Service 4684c1
    /* End of list reached.  */
Packit Service 4684c1
    return primary (fd, request, arg);
Packit Service 4684c1
  else
Packit Service 4684c1
    return remaining_list->private_ioctl_fn (remaining_list->private_next,
Packit Service 4684c1
                                             primary, fd, request, arg);
Packit Service 4684c1
}
Packit Service 4684c1
Packit Service 4684c1
int
Packit Service 4684c1
execute_all_ioctl_hooks (gl_ioctl_fn primary,
Packit Service 4684c1
                         int fd, int request, void *arg)
Packit Service 4684c1
{
Packit Service 4684c1
  return execute_ioctl_hooks (anchor.private_next, primary, fd, request, arg);
Packit Service 4684c1
}
Packit Service 4684c1
Packit Service 4684c1
void
Packit Service 4684c1
register_fd_hook (close_hook_fn close_hook, ioctl_hook_fn ioctl_hook, struct fd_hook *link)
Packit Service 4684c1
{
Packit Service 4684c1
  if (close_hook == NULL)
Packit Service 4684c1
    close_hook = execute_close_hooks;
Packit Service 4684c1
  if (ioctl_hook == NULL)
Packit Service 4684c1
    ioctl_hook = execute_ioctl_hooks;
Packit Service 4684c1
Packit Service 4684c1
  if (link->private_next == NULL && link->private_prev == NULL)
Packit Service 4684c1
    {
Packit Service 4684c1
      /* Add the link to the doubly linked list.  */
Packit Service 4684c1
      link->private_next = anchor.private_next;
Packit Service 4684c1
      link->private_prev = &anchor;
Packit Service 4684c1
      link->private_close_fn = close_hook;
Packit Service 4684c1
      link->private_ioctl_fn = ioctl_hook;
Packit Service 4684c1
      anchor.private_next->private_prev = link;
Packit Service 4684c1
      anchor.private_next = link;
Packit Service 4684c1
    }
Packit Service 4684c1
  else
Packit Service 4684c1
    {
Packit Service 4684c1
      /* The link is already in use.  */
Packit Service 4684c1
      if (link->private_close_fn != close_hook
Packit Service 4684c1
          || link->private_ioctl_fn != ioctl_hook)
Packit Service 4684c1
        abort ();
Packit Service 4684c1
    }
Packit Service 4684c1
}
Packit Service 4684c1
Packit Service 4684c1
void
Packit Service 4684c1
unregister_fd_hook (struct fd_hook *link)
Packit Service 4684c1
{
Packit Service 4684c1
  struct fd_hook *next = link->private_next;
Packit Service 4684c1
  struct fd_hook *prev = link->private_prev;
Packit Service 4684c1
Packit Service 4684c1
  if (next != NULL && prev != NULL)
Packit Service 4684c1
    {
Packit Service 4684c1
      /* The link is in use.  Remove it from the doubly linked list.  */
Packit Service 4684c1
      prev->private_next = next;
Packit Service 4684c1
      next->private_prev = prev;
Packit Service 4684c1
      /* Clear the link, to mark it unused.  */
Packit Service 4684c1
      link->private_next = NULL;
Packit Service 4684c1
      link->private_prev = NULL;
Packit Service 4684c1
      link->private_close_fn = NULL;
Packit Service 4684c1
      link->private_ioctl_fn = NULL;
Packit Service 4684c1
    }
Packit Service 4684c1
}
Packit Service 4684c1
Packit Service 4684c1
#endif