Blame hesiod/nss_hesiod/hesiod-grp.c

Packit Service 82fcde
/* Copyright (C) 1997-2018 Free Software Foundation, Inc.
Packit Service 82fcde
   This file is part of the GNU C Library.
Packit Service 82fcde
   Contributed by Mark Kettenis <kettenis@phys.uva.nl>, 1997.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is free software; you can redistribute it and/or
Packit Service 82fcde
   modify it under the terms of the GNU Lesser General Public
Packit Service 82fcde
   License as published by the Free Software Foundation; either
Packit Service 82fcde
   version 2.1 of the License, or (at your option) any later version.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is distributed in the hope that it will be useful,
Packit Service 82fcde
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 82fcde
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 82fcde
   Lesser General Public License for more details.
Packit Service 82fcde
Packit Service 82fcde
   You should have received a copy of the GNU Lesser General Public
Packit Service 82fcde
   License along with the GNU C Library; if not, see
Packit Service 82fcde
   <http://www.gnu.org/licenses/>.  */
Packit Service 82fcde
Packit Service 82fcde
#include <ctype.h>
Packit Service 82fcde
#include <errno.h>
Packit Service 82fcde
#include <grp.h>
Packit Service 82fcde
#include <hesiod.h>
Packit Service 82fcde
#include <nss.h>
Packit Service 82fcde
#include <stdio.h>
Packit Service 82fcde
#include <stdlib.h>
Packit Service 82fcde
#include <string.h>
Packit Service 82fcde
#include <sys/param.h>
Packit Service 82fcde
Packit Service 82fcde
/* Get the declaration of the parser function.  */
Packit Service 82fcde
#define ENTNAME grent
Packit Service 82fcde
#define STRUCTURE group
Packit Service 82fcde
#define EXTERN_PARSER
Packit Service 82fcde
#include <nss/nss_files/files-parse.c>
Packit Service 82fcde
Packit Service 82fcde
enum nss_status
Packit Service 82fcde
_nss_hesiod_setgrent (int stayopen)
Packit Service 82fcde
{
Packit Service 82fcde
  return NSS_STATUS_SUCCESS;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
enum nss_status
Packit Service 82fcde
_nss_hesiod_endgrent (void)
Packit Service 82fcde
{
Packit Service 82fcde
  return NSS_STATUS_SUCCESS;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
static enum nss_status
Packit Service 82fcde
lookup (const char *name, const char *type, struct group *grp,
Packit Service 82fcde
	char *buffer, size_t buflen, int *errnop)
Packit Service 82fcde
{
Packit Service 82fcde
  struct parser_data *data = (void *) buffer;
Packit Service 82fcde
  size_t linebuflen;
Packit Service 82fcde
  void *context;
Packit Service 82fcde
  char **list;
Packit Service 82fcde
  int parse_res;
Packit Service 82fcde
  size_t len;
Packit Service 82fcde
  int olderr = errno;
Packit Service 82fcde
Packit Service 82fcde
  if (hesiod_init (&context) < 0)
Packit Service 82fcde
    return NSS_STATUS_UNAVAIL;
Packit Service 82fcde
Packit Service 82fcde
  list = hesiod_resolve (context, name, type);
Packit Service 82fcde
  if (list == NULL)
Packit Service 82fcde
    {
Packit Service 82fcde
      int err = errno;
Packit Service 82fcde
      hesiod_end (context);
Packit Service 82fcde
      __set_errno (olderr);
Packit Service 82fcde
      return err == ENOENT ? NSS_STATUS_NOTFOUND : NSS_STATUS_UNAVAIL;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  linebuflen = buffer + buflen - data->linebuffer;
Packit Service 82fcde
  len = strlen (*list) + 1;
Packit Service 82fcde
  if (linebuflen < len)
Packit Service 82fcde
    {
Packit Service 82fcde
      hesiod_free_list (context, list);
Packit Service 82fcde
      hesiod_end (context);
Packit Service 82fcde
      *errnop = ERANGE;
Packit Service 82fcde
      return NSS_STATUS_TRYAGAIN;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  memcpy (data->linebuffer, *list, len);
Packit Service 82fcde
  hesiod_free_list (context, list);
Packit Service 82fcde
  hesiod_end (context);
Packit Service 82fcde
Packit Service 82fcde
  parse_res = _nss_files_parse_grent (buffer, grp, data, buflen, errnop);
Packit Service 82fcde
  if (parse_res < 1)
Packit Service 82fcde
    {
Packit Service 82fcde
      __set_errno (olderr);
Packit Service 82fcde
      return parse_res == -1 ? NSS_STATUS_TRYAGAIN : NSS_STATUS_NOTFOUND;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  return NSS_STATUS_SUCCESS;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
enum nss_status
Packit Service 82fcde
_nss_hesiod_getgrnam_r (const char *name, struct group *grp,
Packit Service 82fcde
			char *buffer, size_t buflen, int *errnop)
Packit Service 82fcde
{
Packit Service 82fcde
  return lookup (name, "group", grp, buffer, buflen, errnop);
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
enum nss_status
Packit Service 82fcde
_nss_hesiod_getgrgid_r (gid_t gid, struct group *grp,
Packit Service 82fcde
			char *buffer, size_t buflen, int *errnop)
Packit Service 82fcde
{
Packit Service 82fcde
  char gidstr[21];	/* We will probably never have a gid_t with more
Packit Service 82fcde
			   than 64 bits.  */
Packit Service 82fcde
Packit Service 82fcde
  snprintf (gidstr, sizeof gidstr, "%d", gid);
Packit Service 82fcde
Packit Service 82fcde
  return lookup (gidstr, "gid", grp, buffer, buflen, errnop);
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
static int
Packit Service 82fcde
internal_gid_in_list (const gid_t *list, const gid_t g, long int len)
Packit Service 82fcde
{
Packit Service 82fcde
  while (len > 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      if (*list == g)
Packit Service 82fcde
	return 1;
Packit Service 82fcde
      --len;
Packit Service 82fcde
      ++list;
Packit Service 82fcde
    }
Packit Service 82fcde
  return 0;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
static enum nss_status
Packit Service 82fcde
internal_gid_from_group (void *context, const char *groupname, gid_t *group)
Packit Service 82fcde
{
Packit Service 82fcde
  char **grp_res;
Packit Service 82fcde
  enum nss_status status = NSS_STATUS_NOTFOUND;
Packit Service 82fcde
Packit Service 82fcde
  grp_res = hesiod_resolve (context, groupname, "group");
Packit Service 82fcde
  if (grp_res != NULL && *grp_res != NULL)
Packit Service 82fcde
    {
Packit Service 82fcde
      char *p = *grp_res;
Packit Service 82fcde
Packit Service 82fcde
      /* Skip to third field.  */
Packit Service 82fcde
      while (*p != '\0' && *p != ':')
Packit Service 82fcde
	++p;
Packit Service 82fcde
      if (*p != '\0')
Packit Service 82fcde
	++p;
Packit Service 82fcde
      while (*p != '\0' && *p != ':')
Packit Service 82fcde
	++p;
Packit Service 82fcde
      if (*p != '\0')
Packit Service 82fcde
	{
Packit Service 82fcde
	  char *endp;
Packit Service 82fcde
	  char *q = ++p;
Packit Service 82fcde
	  long int val;
Packit Service 82fcde
Packit Service 82fcde
	  while (*q != '\0' && *q != ':')
Packit Service 82fcde
	    ++q;
Packit Service 82fcde
Packit Service 82fcde
	  val = strtol (p, &endp, 10);
Packit Service 82fcde
	  if (sizeof (gid_t) == sizeof (long int) || (gid_t) val == val)
Packit Service 82fcde
	    {
Packit Service 82fcde
	      *group = val;
Packit Service 82fcde
	      if (endp == q && endp != p)
Packit Service 82fcde
		status = NSS_STATUS_SUCCESS;
Packit Service 82fcde
	    }
Packit Service 82fcde
        }
Packit Service 82fcde
      hesiod_free_list (context, grp_res);
Packit Service 82fcde
    }
Packit Service 82fcde
  return status;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
enum nss_status
Packit Service 82fcde
_nss_hesiod_initgroups_dyn (const char *user, gid_t group, long int *start,
Packit Service 82fcde
			    long int *size, gid_t **groupsp, long int limit,
Packit Service 82fcde
			    int *errnop)
Packit Service 82fcde
{
Packit Service 82fcde
  enum nss_status status = NSS_STATUS_SUCCESS;
Packit Service 82fcde
  char **list = NULL;
Packit Service 82fcde
  char *p;
Packit Service 82fcde
  void *context;
Packit Service 82fcde
  gid_t *groups = *groupsp;
Packit Service 82fcde
  int save_errno;
Packit Service 82fcde
Packit Service 82fcde
  if (hesiod_init (&context) < 0)
Packit Service 82fcde
    return NSS_STATUS_UNAVAIL;
Packit Service 82fcde
Packit Service 82fcde
  list = hesiod_resolve (context, user, "grplist");
Packit Service 82fcde
Packit Service 82fcde
  if (list == NULL)
Packit Service 82fcde
    {
Packit Service 82fcde
      hesiod_end (context);
Packit Service 82fcde
      return errno == ENOENT ? NSS_STATUS_NOTFOUND : NSS_STATUS_UNAVAIL;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  save_errno = errno;
Packit Service 82fcde
Packit Service 82fcde
  p = *list;
Packit Service 82fcde
  while (*p != '\0')
Packit Service 82fcde
    {
Packit Service 82fcde
      char *endp;
Packit Service 82fcde
      char *q;
Packit Service 82fcde
      long int val;
Packit Service 82fcde
Packit Service 82fcde
      status = NSS_STATUS_NOTFOUND;
Packit Service 82fcde
Packit Service 82fcde
      q = p;
Packit Service 82fcde
      while (*q != '\0' && *q != ':' && *q != ',')
Packit Service 82fcde
	++q;
Packit Service 82fcde
Packit Service 82fcde
      if (*q != '\0')
Packit Service 82fcde
	*q++ = '\0';
Packit Service 82fcde
Packit Service 82fcde
      __set_errno (0);
Packit Service 82fcde
      val = strtol (p, &endp, 10);
Packit Service 82fcde
      /* Test whether the number is representable in a variable of
Packit Service 82fcde
         type `gid_t'.  If not ignore the number.  */
Packit Service 82fcde
      if ((sizeof (gid_t) == sizeof (long int) || (gid_t) val == val)
Packit Service 82fcde
	  && errno == 0)
Packit Service 82fcde
	{
Packit Service 82fcde
	  if (*endp == '\0' && endp != p)
Packit Service 82fcde
	    {
Packit Service 82fcde
	      group = val;
Packit Service 82fcde
	      status = NSS_STATUS_SUCCESS;
Packit Service 82fcde
	    }
Packit Service 82fcde
	  else
Packit Service 82fcde
	    status = internal_gid_from_group (context, p, &group);
Packit Service 82fcde
Packit Service 82fcde
	  if (status == NSS_STATUS_SUCCESS
Packit Service 82fcde
	      && !internal_gid_in_list (groups, group, *start))
Packit Service 82fcde
	    {
Packit Service 82fcde
	      if (__glibc_unlikely (*start == *size))
Packit Service 82fcde
		{
Packit Service 82fcde
		  /* Need a bigger buffer.  */
Packit Service 82fcde
		  gid_t *newgroups;
Packit Service 82fcde
		  long int newsize;
Packit Service 82fcde
Packit Service 82fcde
		  if (limit > 0 && *size == limit)
Packit Service 82fcde
		    /* We reached the maximum.  */
Packit Service 82fcde
		    goto done;
Packit Service 82fcde
Packit Service 82fcde
		  if (limit <= 0)
Packit Service 82fcde
		    newsize = 2 * *size;
Packit Service 82fcde
		  else
Packit Service 82fcde
		    newsize = MIN (limit, 2 * *size);
Packit Service 82fcde
Packit Service 82fcde
		  newgroups = realloc (groups, newsize * sizeof (*groups));
Packit Service 82fcde
		  if (newgroups == NULL)
Packit Service 82fcde
		    goto done;
Packit Service 82fcde
		  *groupsp = groups = newgroups;
Packit Service 82fcde
		  *size = newsize;
Packit Service 82fcde
		}
Packit Service 82fcde
Packit Service 82fcde
	      groups[(*start)++] = group;
Packit Service 82fcde
	    }
Packit Service 82fcde
	}
Packit Service 82fcde
Packit Service 82fcde
      p = q;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  __set_errno (save_errno);
Packit Service 82fcde
Packit Service 82fcde
 done:
Packit Service 82fcde
  hesiod_free_list (context, list);
Packit Service 82fcde
  hesiod_end (context);
Packit Service 82fcde
Packit Service 82fcde
  return NSS_STATUS_SUCCESS;
Packit Service 82fcde
}