mirror of
https://github.com/Stichting-MINIX-Research-Foundation/netbsd.git
synced 2025-08-18 10:31:01 -04:00
222 lines
4.8 KiB
C
222 lines
4.8 KiB
C
/* $NetBSD: getnetgrent_r.c,v 1.1.1.2 2012/09/09 16:07:57 christos Exp $ */
|
|
|
|
/*
|
|
* Copyright (C) 2004, 2005, 2008 Internet Systems Consortium, Inc. ("ISC")
|
|
* Copyright (C) 1998, 1999, 2001, 2003 Internet Software Consortium.
|
|
*
|
|
* Permission to use, copy, modify, and/or distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice appear in all copies.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
|
|
* REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
|
* AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
|
|
* INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
|
* LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
|
|
* OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
* PERFORMANCE OF THIS SOFTWARE.
|
|
*/
|
|
|
|
#if defined(LIBC_SCCS) && !defined(lint)
|
|
static const char rcsid[] = "Id: getnetgrent_r.c,v 1.14 2008/11/14 02:36:51 marka Exp ";
|
|
#endif /* LIBC_SCCS and not lint */
|
|
|
|
#include <port_before.h>
|
|
#if !defined(_REENTRANT) || !defined(DO_PTHREADS)
|
|
static int getnetgrent_r_not_required = 0;
|
|
#else
|
|
#include <errno.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <sys/types.h>
|
|
#include <netinet/in.h>
|
|
#include <netdb.h>
|
|
#include <stdlib.h>
|
|
#include <port_after.h>
|
|
|
|
#ifdef NGR_R_RETURN
|
|
#ifndef NGR_R_PRIVATE
|
|
#define NGR_R_PRIVATE 0
|
|
#endif
|
|
|
|
static NGR_R_RETURN
|
|
copy_protoent(NGR_R_CONST char **, NGR_R_CONST char **, NGR_R_CONST char **,
|
|
const char *, const char *, const char *, NGR_R_COPY_ARGS);
|
|
|
|
NGR_R_RETURN
|
|
innetgr_r(const char *netgroup, const char *host, const char *user,
|
|
const char *domain) {
|
|
char *ng, *ho, *us, *dom;
|
|
|
|
DE_CONST(netgroup, ng);
|
|
DE_CONST(host, ho);
|
|
DE_CONST(user, us);
|
|
DE_CONST(domain, dom);
|
|
|
|
return (innetgr(ng, ho, us, dom));
|
|
}
|
|
|
|
/*%
|
|
* These assume a single context is in operation per thread.
|
|
* If this is not the case we will need to call irs directly
|
|
* rather than through the base functions.
|
|
*/
|
|
|
|
NGR_R_RETURN
|
|
getnetgrent_r(NGR_R_CONST char **machinep, NGR_R_CONST char **userp,
|
|
NGR_R_CONST char **domainp, NGR_R_ARGS)
|
|
{
|
|
NGR_R_CONST char *mp, *up, *dp;
|
|
int res = getnetgrent(&mp, &up, &dp);
|
|
|
|
if (res != 1)
|
|
return (res);
|
|
|
|
return (copy_protoent(machinep, userp, domainp,
|
|
mp, up, dp, NGR_R_COPY));
|
|
}
|
|
|
|
#if NGR_R_PRIVATE == 2
|
|
struct private {
|
|
char *buf;
|
|
};
|
|
|
|
#endif
|
|
NGR_R_SET_RETURN
|
|
#ifdef NGR_R_SET_ARGS
|
|
setnetgrent_r(NGR_R_SET_CONST char *netgroup, NGR_R_SET_ARGS)
|
|
#else
|
|
setnetgrent_r(NGR_R_SET_CONST char *netgroup)
|
|
#endif
|
|
{
|
|
#if NGR_R_PRIVATE == 2
|
|
struct private *p;
|
|
#endif
|
|
char *tmp;
|
|
#if defined(NGR_R_SET_ARGS) && NGR_R_PRIVATE == 0
|
|
UNUSED(buf);
|
|
UNUSED(buflen);
|
|
#endif
|
|
|
|
DE_CONST(netgroup, tmp);
|
|
setnetgrent(tmp);
|
|
|
|
#if NGR_R_PRIVATE == 1
|
|
*buf = NULL;
|
|
#elif NGR_R_PRIVATE == 2
|
|
*buf = p = malloc(sizeof(struct private));
|
|
if (p == NULL)
|
|
#ifdef NGR_R_SET_RESULT
|
|
return (NGR_R_BAD);
|
|
#else
|
|
return;
|
|
#endif
|
|
p->buf = NULL;
|
|
#endif
|
|
#ifdef NGR_R_SET_RESULT
|
|
return (NGR_R_SET_RESULT);
|
|
#endif
|
|
}
|
|
|
|
NGR_R_END_RETURN
|
|
#ifdef NGR_R_END_ARGS
|
|
endnetgrent_r(NGR_R_END_ARGS)
|
|
#else
|
|
endnetgrent_r(void)
|
|
#endif
|
|
{
|
|
#if NGR_R_PRIVATE == 2
|
|
struct private *p = buf;
|
|
#endif
|
|
#if defined(NGR_R_SET_ARGS) && NGR_R_PRIVATE == 0
|
|
UNUSED(buf);
|
|
UNUSED(buflen);
|
|
#endif
|
|
|
|
endnetgrent();
|
|
#if NGR_R_PRIVATE == 1
|
|
if (*buf != NULL)
|
|
free(*buf);
|
|
*buf = NULL;
|
|
#elif NGR_R_PRIVATE == 2
|
|
if (p->buf != NULL)
|
|
free(p->buf);
|
|
free(p);
|
|
#endif
|
|
NGR_R_END_RESULT(NGR_R_OK);
|
|
}
|
|
|
|
/* Private */
|
|
|
|
static int
|
|
copy_protoent(NGR_R_CONST char **machinep, NGR_R_CONST char **userp,
|
|
NGR_R_CONST char **domainp, const char *mp, const char *up,
|
|
const char *dp, NGR_R_COPY_ARGS)
|
|
{
|
|
#if NGR_R_PRIVATE == 2
|
|
struct private *p = buf;
|
|
#endif
|
|
char *cp;
|
|
int n;
|
|
int len;
|
|
|
|
/* Find out the amount of space required to store the answer. */
|
|
len = 0;
|
|
if (mp != NULL) len += strlen(mp) + 1;
|
|
if (up != NULL) len += strlen(up) + 1;
|
|
if (dp != NULL) len += strlen(dp) + 1;
|
|
|
|
#if NGR_R_PRIVATE == 1
|
|
if (*buf != NULL)
|
|
free(*buf);
|
|
*buf = malloc(len);
|
|
if (*buf == NULL)
|
|
return(NGR_R_BAD);
|
|
cp = *buf;
|
|
#elif NGR_R_PRIVATE == 2
|
|
if (p->buf)
|
|
free(p->buf);
|
|
p->buf = malloc(len);
|
|
if (p->buf == NULL)
|
|
return(NGR_R_BAD);
|
|
cp = p->buf;
|
|
#else
|
|
if (len > (int)buflen) {
|
|
errno = ERANGE;
|
|
return (NGR_R_BAD);
|
|
}
|
|
cp = buf;
|
|
#endif
|
|
|
|
if (mp != NULL) {
|
|
n = strlen(mp) + 1;
|
|
strcpy(cp, mp);
|
|
*machinep = cp;
|
|
cp += n;
|
|
} else
|
|
*machinep = NULL;
|
|
|
|
if (up != NULL) {
|
|
n = strlen(up) + 1;
|
|
strcpy(cp, up);
|
|
*userp = cp;
|
|
cp += n;
|
|
} else
|
|
*userp = NULL;
|
|
|
|
if (dp != NULL) {
|
|
n = strlen(dp) + 1;
|
|
strcpy(cp, dp);
|
|
*domainp = cp;
|
|
cp += n;
|
|
} else
|
|
*domainp = NULL;
|
|
|
|
return (NGR_R_OK);
|
|
}
|
|
#else /* NGR_R_RETURN */
|
|
static int getnetgrent_r_unknown_system = 0;
|
|
#endif /* NGR_R_RETURN */
|
|
#endif /* !defined(_REENTRANT) || !defined(DO_PTHREADS) */
|
|
/*! \file */
|