Blame sysdeps/unix/sysv/linux/getsourcefilter.c

Packit 6c4009
/* Get source filter.  Linux version.
Packit 6c4009
   Copyright (C) 2004-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
   Contributed by Ulrich Drepper <drepper@redhat.com>, 2004.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <alloca.h>
Packit 6c4009
#include <assert.h>
Packit 6c4009
#include <errno.h>
Packit 6c4009
#include <stdlib.h>
Packit 6c4009
#include <string.h>
Packit 6c4009
#include <stdint.h>
Packit 6c4009
#include <netatalk/at.h>
Packit 6c4009
#include <netax25/ax25.h>
Packit 6c4009
#include <netinet/in.h>
Packit 6c4009
#include <netipx/ipx.h>
Packit 6c4009
#include <netpacket/packet.h>
Packit 6c4009
#include <netrose/rose.h>
Packit 6c4009
#include <sys/param.h>
Packit 6c4009
#include <sys/socket.h>
Packit 6c4009
#include "getsourcefilter.h"
Packit 6c4009
Packit 6c4009
Packit 6c4009
static const struct
Packit 6c4009
{
Packit 6c4009
  int sol;
Packit 6c4009
  int af;
Packit 6c4009
  socklen_t size;
Packit 6c4009
}  sol_map[] =
Packit 6c4009
  {
Packit 6c4009
    /* Sort the array according to importance of the protocols.  Add
Packit 6c4009
       more protocols when they become available.  */
Packit 6c4009
    { SOL_IP, AF_INET, sizeof (struct sockaddr_in) },
Packit 6c4009
    { SOL_IPV6, AF_INET6, sizeof (struct sockaddr_in6) },
Packit 6c4009
    { SOL_AX25, AF_AX25, sizeof (struct sockaddr_ax25) },
Packit 6c4009
    { SOL_IPX, AF_IPX, sizeof (struct sockaddr_ipx) },
Packit 6c4009
    { SOL_ATALK, AF_APPLETALK, sizeof (struct sockaddr_at) },
Packit 6c4009
    { SOL_ROSE, AF_ROSE, sizeof (struct sockaddr_rose) },
Packit 6c4009
    { SOL_PACKET, AF_PACKET, sizeof (struct sockaddr_ll) }
Packit 6c4009
  };
Packit 6c4009
#define NSOL_MAP (sizeof (sol_map) / sizeof (sol_map[0]))
Packit 6c4009
Packit 6c4009
Packit 6c4009
/* Try to determine the socket level value.  Ideally both side and
Packit 6c4009
   family are set.  But sometimes only the size is correct and the
Packit 6c4009
   family value might be bogus.  Loop over the array entries and look
Packit 6c4009
   for a perfect match or the first match based on size.  */
Packit 6c4009
int
Packit 6c4009
__get_sol (int af, socklen_t len)
Packit 6c4009
{
Packit 6c4009
  int first_size_sol = -1;
Packit 6c4009
Packit 6c4009
  for (size_t cnt = 0; cnt < NSOL_MAP; ++cnt)
Packit 6c4009
    {
Packit 6c4009
      /* Just a test so that we make sure the special value used to
Packit 6c4009
	 signal the "we have so far no socket level value" is OK.  */
Packit 6c4009
      assert (sol_map[cnt].sol != -1);
Packit 6c4009
Packit 6c4009
      if (len == sol_map[cnt].size)
Packit 6c4009
	{
Packit 6c4009
	  /* The size matches, which is a requirement.  If the family
Packit 6c4009
	     matches, too, we have a winner.  Otherwise we remember the
Packit 6c4009
	     socket level value for this protocol if it is the first
Packit 6c4009
	     match.  */
Packit 6c4009
	  if (af == sol_map[cnt].af)
Packit 6c4009
	    /* Bingo!  */
Packit 6c4009
	    return sol_map[cnt].sol;
Packit 6c4009
Packit 6c4009
	  if (first_size_sol == -1)
Packit 6c4009
	    first_size_sol = sol_map[cnt].sol;
Packit 6c4009
      }
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  return first_size_sol;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
Packit 6c4009
int
Packit 6c4009
getsourcefilter (int s, uint32_t interface, const struct sockaddr *group,
Packit 6c4009
		 socklen_t grouplen, uint32_t *fmode, uint32_t *numsrc,
Packit 6c4009
		 struct sockaddr_storage *slist)
Packit 6c4009
{
Packit 6c4009
  /* We have to create an struct ip_msfilter object which we can pass
Packit 6c4009
     to the kernel.  */
Packit 6c4009
  socklen_t needed = GROUP_FILTER_SIZE (*numsrc);
Packit 6c4009
  int use_alloca = __libc_use_alloca (needed);
Packit 6c4009
Packit 6c4009
  struct group_filter *gf;
Packit 6c4009
  if (use_alloca)
Packit 6c4009
    gf = (struct group_filter *) alloca (needed);
Packit 6c4009
  else
Packit 6c4009
    {
Packit 6c4009
      gf = (struct group_filter *) malloc (needed);
Packit 6c4009
      if (gf == NULL)
Packit 6c4009
	return -1;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  gf->gf_interface = interface;
Packit 6c4009
  memcpy (&gf->gf_group, group, grouplen);
Packit 6c4009
  gf->gf_numsrc = *numsrc;
Packit 6c4009
Packit 6c4009
  /* We need to provide the appropriate socket level value.  */
Packit 6c4009
  int result;
Packit 6c4009
  int sol = __get_sol (group->sa_family, grouplen);
Packit 6c4009
  if (sol == -1)
Packit 6c4009
    {
Packit 6c4009
      __set_errno (EINVAL);
Packit 6c4009
      result = -1;
Packit 6c4009
    }
Packit 6c4009
  else
Packit 6c4009
    {
Packit 6c4009
      result = __getsockopt (s, sol, MCAST_MSFILTER, gf, &needed);
Packit 6c4009
Packit 6c4009
      /* If successful, copy the results to the places the caller wants
Packit 6c4009
	 them in.  */
Packit 6c4009
      if (result == 0)
Packit 6c4009
	{
Packit 6c4009
	  *fmode = gf->gf_fmode;
Packit 6c4009
	  memcpy (slist, gf->gf_slist,
Packit 6c4009
		  MIN (*numsrc, gf->gf_numsrc)
Packit 6c4009
		  * sizeof (struct sockaddr_storage));
Packit 6c4009
	  *numsrc = gf->gf_numsrc;
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  if (! use_alloca)
Packit 6c4009
    {
Packit 6c4009
      int save_errno = errno;
Packit 6c4009
      free (gf);
Packit 6c4009
      __set_errno (save_errno);
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  return result;
Packit 6c4009
}