Blame hesiod/nss_hesiod/hesiod-grp.c

Packit 6c4009
/* Copyright (C) 1997-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
   Contributed by Mark Kettenis <kettenis@phys.uva.nl>, 1997.
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 <ctype.h>
Packit 6c4009
#include <errno.h>
Packit 6c4009
#include <grp.h>
Packit 6c4009
#include <hesiod.h>
Packit 6c4009
#include <nss.h>
Packit 6c4009
#include <stdio.h>
Packit 6c4009
#include <stdlib.h>
Packit 6c4009
#include <string.h>
Packit 6c4009
#include <sys/param.h>
Packit 6c4009
Packit 6c4009
/* Get the declaration of the parser function.  */
Packit 6c4009
#define ENTNAME grent
Packit 6c4009
#define STRUCTURE group
Packit 6c4009
#define EXTERN_PARSER
Packit 6c4009
#include <nss/nss_files/files-parse.c>
Packit 6c4009
Packit 6c4009
enum nss_status
Packit 6c4009
_nss_hesiod_setgrent (int stayopen)
Packit 6c4009
{
Packit 6c4009
  return NSS_STATUS_SUCCESS;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
enum nss_status
Packit 6c4009
_nss_hesiod_endgrent (void)
Packit 6c4009
{
Packit 6c4009
  return NSS_STATUS_SUCCESS;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
static enum nss_status
Packit 6c4009
lookup (const char *name, const char *type, struct group *grp,
Packit 6c4009
	char *buffer, size_t buflen, int *errnop)
Packit 6c4009
{
Packit 6c4009
  struct parser_data *data = (void *) buffer;
Packit 6c4009
  size_t linebuflen;
Packit 6c4009
  void *context;
Packit 6c4009
  char **list;
Packit 6c4009
  int parse_res;
Packit 6c4009
  size_t len;
Packit 6c4009
  int olderr = errno;
Packit 6c4009
Packit 6c4009
  if (hesiod_init (&context) < 0)
Packit 6c4009
    return NSS_STATUS_UNAVAIL;
Packit 6c4009
Packit 6c4009
  list = hesiod_resolve (context, name, type);
Packit 6c4009
  if (list == NULL)
Packit 6c4009
    {
Packit 6c4009
      int err = errno;
Packit 6c4009
      hesiod_end (context);
Packit 6c4009
      __set_errno (olderr);
Packit 6c4009
      return err == ENOENT ? NSS_STATUS_NOTFOUND : NSS_STATUS_UNAVAIL;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  linebuflen = buffer + buflen - data->linebuffer;
Packit 6c4009
  len = strlen (*list) + 1;
Packit 6c4009
  if (linebuflen < len)
Packit 6c4009
    {
Packit 6c4009
      hesiod_free_list (context, list);
Packit 6c4009
      hesiod_end (context);
Packit 6c4009
      *errnop = ERANGE;
Packit 6c4009
      return NSS_STATUS_TRYAGAIN;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  memcpy (data->linebuffer, *list, len);
Packit 6c4009
  hesiod_free_list (context, list);
Packit 6c4009
  hesiod_end (context);
Packit 6c4009
Packit 6c4009
  parse_res = _nss_files_parse_grent (buffer, grp, data, buflen, errnop);
Packit 6c4009
  if (parse_res < 1)
Packit 6c4009
    {
Packit 6c4009
      __set_errno (olderr);
Packit 6c4009
      return parse_res == -1 ? NSS_STATUS_TRYAGAIN : NSS_STATUS_NOTFOUND;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  return NSS_STATUS_SUCCESS;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
enum nss_status
Packit 6c4009
_nss_hesiod_getgrnam_r (const char *name, struct group *grp,
Packit 6c4009
			char *buffer, size_t buflen, int *errnop)
Packit 6c4009
{
Packit 6c4009
  return lookup (name, "group", grp, buffer, buflen, errnop);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
enum nss_status
Packit 6c4009
_nss_hesiod_getgrgid_r (gid_t gid, struct group *grp,
Packit 6c4009
			char *buffer, size_t buflen, int *errnop)
Packit 6c4009
{
Packit 6c4009
  char gidstr[21];	/* We will probably never have a gid_t with more
Packit 6c4009
			   than 64 bits.  */
Packit 6c4009
Packit 6c4009
  snprintf (gidstr, sizeof gidstr, "%d", gid);
Packit 6c4009
Packit 6c4009
  return lookup (gidstr, "gid", grp, buffer, buflen, errnop);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
static int
Packit 6c4009
internal_gid_in_list (const gid_t *list, const gid_t g, long int len)
Packit 6c4009
{
Packit 6c4009
  while (len > 0)
Packit 6c4009
    {
Packit 6c4009
      if (*list == g)
Packit 6c4009
	return 1;
Packit 6c4009
      --len;
Packit 6c4009
      ++list;
Packit 6c4009
    }
Packit 6c4009
  return 0;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
static enum nss_status
Packit 6c4009
internal_gid_from_group (void *context, const char *groupname, gid_t *group)
Packit 6c4009
{
Packit 6c4009
  char **grp_res;
Packit 6c4009
  enum nss_status status = NSS_STATUS_NOTFOUND;
Packit 6c4009
Packit 6c4009
  grp_res = hesiod_resolve (context, groupname, "group");
Packit 6c4009
  if (grp_res != NULL && *grp_res != NULL)
Packit 6c4009
    {
Packit 6c4009
      char *p = *grp_res;
Packit 6c4009
Packit 6c4009
      /* Skip to third field.  */
Packit 6c4009
      while (*p != '\0' && *p != ':')
Packit 6c4009
	++p;
Packit 6c4009
      if (*p != '\0')
Packit 6c4009
	++p;
Packit 6c4009
      while (*p != '\0' && *p != ':')
Packit 6c4009
	++p;
Packit 6c4009
      if (*p != '\0')
Packit 6c4009
	{
Packit 6c4009
	  char *endp;
Packit 6c4009
	  char *q = ++p;
Packit 6c4009
	  long int val;
Packit 6c4009
Packit 6c4009
	  while (*q != '\0' && *q != ':')
Packit 6c4009
	    ++q;
Packit 6c4009
Packit 6c4009
	  val = strtol (p, &endp, 10);
Packit 6c4009
	  if (sizeof (gid_t) == sizeof (long int) || (gid_t) val == val)
Packit 6c4009
	    {
Packit 6c4009
	      *group = val;
Packit 6c4009
	      if (endp == q && endp != p)
Packit 6c4009
		status = NSS_STATUS_SUCCESS;
Packit 6c4009
	    }
Packit 6c4009
        }
Packit 6c4009
      hesiod_free_list (context, grp_res);
Packit 6c4009
    }
Packit 6c4009
  return status;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
enum nss_status
Packit 6c4009
_nss_hesiod_initgroups_dyn (const char *user, gid_t group, long int *start,
Packit 6c4009
			    long int *size, gid_t **groupsp, long int limit,
Packit 6c4009
			    int *errnop)
Packit 6c4009
{
Packit 6c4009
  enum nss_status status = NSS_STATUS_SUCCESS;
Packit 6c4009
  char **list = NULL;
Packit 6c4009
  char *p;
Packit 6c4009
  void *context;
Packit 6c4009
  gid_t *groups = *groupsp;
Packit 6c4009
  int save_errno;
Packit 6c4009
Packit 6c4009
  if (hesiod_init (&context) < 0)
Packit 6c4009
    return NSS_STATUS_UNAVAIL;
Packit 6c4009
Packit 6c4009
  list = hesiod_resolve (context, user, "grplist");
Packit 6c4009
Packit 6c4009
  if (list == NULL)
Packit 6c4009
    {
Packit 6c4009
      hesiod_end (context);
Packit 6c4009
      return errno == ENOENT ? NSS_STATUS_NOTFOUND : NSS_STATUS_UNAVAIL;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  save_errno = errno;
Packit 6c4009
Packit 6c4009
  p = *list;
Packit 6c4009
  while (*p != '\0')
Packit 6c4009
    {
Packit 6c4009
      char *endp;
Packit 6c4009
      char *q;
Packit 6c4009
      long int val;
Packit 6c4009
Packit 6c4009
      status = NSS_STATUS_NOTFOUND;
Packit 6c4009
Packit 6c4009
      q = p;
Packit 6c4009
      while (*q != '\0' && *q != ':' && *q != ',')
Packit 6c4009
	++q;
Packit 6c4009
Packit 6c4009
      if (*q != '\0')
Packit 6c4009
	*q++ = '\0';
Packit 6c4009
Packit 6c4009
      __set_errno (0);
Packit 6c4009
      val = strtol (p, &endp, 10);
Packit 6c4009
      /* Test whether the number is representable in a variable of
Packit 6c4009
         type `gid_t'.  If not ignore the number.  */
Packit 6c4009
      if ((sizeof (gid_t) == sizeof (long int) || (gid_t) val == val)
Packit 6c4009
	  && errno == 0)
Packit 6c4009
	{
Packit 6c4009
	  if (*endp == '\0' && endp != p)
Packit 6c4009
	    {
Packit 6c4009
	      group = val;
Packit 6c4009
	      status = NSS_STATUS_SUCCESS;
Packit 6c4009
	    }
Packit 6c4009
	  else
Packit 6c4009
	    status = internal_gid_from_group (context, p, &group);
Packit 6c4009
Packit 6c4009
	  if (status == NSS_STATUS_SUCCESS
Packit 6c4009
	      && !internal_gid_in_list (groups, group, *start))
Packit 6c4009
	    {
Packit 6c4009
	      if (__glibc_unlikely (*start == *size))
Packit 6c4009
		{
Packit 6c4009
		  /* Need a bigger buffer.  */
Packit 6c4009
		  gid_t *newgroups;
Packit 6c4009
		  long int newsize;
Packit 6c4009
Packit 6c4009
		  if (limit > 0 && *size == limit)
Packit 6c4009
		    /* We reached the maximum.  */
Packit 6c4009
		    goto done;
Packit 6c4009
Packit 6c4009
		  if (limit <= 0)
Packit 6c4009
		    newsize = 2 * *size;
Packit 6c4009
		  else
Packit 6c4009
		    newsize = MIN (limit, 2 * *size);
Packit 6c4009
Packit 6c4009
		  newgroups = realloc (groups, newsize * sizeof (*groups));
Packit 6c4009
		  if (newgroups == NULL)
Packit 6c4009
		    goto done;
Packit 6c4009
		  *groupsp = groups = newgroups;
Packit 6c4009
		  *size = newsize;
Packit 6c4009
		}
Packit 6c4009
Packit 6c4009
	      groups[(*start)++] = group;
Packit 6c4009
	    }
Packit 6c4009
	}
Packit 6c4009
Packit 6c4009
      p = q;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  __set_errno (save_errno);
Packit 6c4009
Packit 6c4009
 done:
Packit 6c4009
  hesiod_free_list (context, list);
Packit 6c4009
  hesiod_end (context);
Packit 6c4009
Packit 6c4009
  return NSS_STATUS_SUCCESS;
Packit 6c4009
}