| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288 | /*** services.c                           /etc/services access functions**** This file is part of the NYS Library.**** The NYS Library is free software; you can redistribute it and/or** modify it under the terms of the GNU Library General Public License as** published by the Free Software Foundation; either version 2 of the** License, or (at your option) any later version.**** The NYS Library is distributed in the hope that it will be useful,** but WITHOUT ANY WARRANTY; without even the implied warranty of** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU** Library General Public License for more details.**** You should have received a copy of the GNU Library General Public** License along with the NYS Library; see the file COPYING.LIB.  If** not, write to the Free Software Foundation, Inc., 675 Mass Ave,** Cambridge, MA 02139, USA.****** Copyright (c) 1983 Regents of the University of California.** All rights reserved.**** Redistribution and use in source and binary forms, with or without** modification, are permitted provided that the following conditions** are met:** 1. Redistributions of source code must retain the above copyright**    notice, this list of conditions and the following disclaimer.** 2. Redistributions in binary form must reproduce the above copyright**    notice, this list of conditions and the following disclaimer in the**    documentation and/or other materials provided with the distribution.** 3. All advertising materials mentioning features or use of this software**    must display the following acknowledgement:**	This product includes software developed by the University of**	California, Berkeley and its contributors.** 4. Neither the name of the University nor the names of its contributors**    may be used to endorse or promote products derived from this software**    without specific prior written permission.**** THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND** ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE** IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE** ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE** FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL** DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS** OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)** HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT** LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY** OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF** SUCH DAMAGE.*/#define __FORCE_GLIBC#include <features.h>#include <sys/types.h>#include <sys/socket.h>#include <netdb.h>#include <stdio.h>#include <string.h>#include <stdlib.h>#include <netinet/in.h>#include <arpa/inet.h>#include <errno.h>libc_hidden_proto(strcmp)libc_hidden_proto(strpbrk)libc_hidden_proto(fopen)libc_hidden_proto(fclose)libc_hidden_proto(atoi)libc_hidden_proto(rewind)libc_hidden_proto(fgets)libc_hidden_proto(abort)#ifdef __UCLIBC_HAS_THREADS__# include <pthread.h>static pthread_mutex_t mylock = PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP;#endif#define LOCK	__pthread_mutex_lock(&mylock)#define UNLOCK	__pthread_mutex_unlock(&mylock)#define	MAXALIASES	35#define SBUFSIZE	(BUFSIZ + 1 + (sizeof(char *) * MAXALIASES))static FILE *servf = NULL;static struct servent serv;static char *servbuf = NULL;static int serv_stayopen;static void __initbuf(void){    if (!servbuf) {	servbuf = malloc(SBUFSIZE);	if (!servbuf)	    abort();    }}libc_hidden_proto(setservent)void setservent(int f){    LOCK;    if (servf == NULL)	servf = fopen(_PATH_SERVICES, "r" );    else	rewind(servf);    serv_stayopen |= f;    UNLOCK;}libc_hidden_def(setservent)libc_hidden_proto(endservent)void endservent(void){    LOCK;    if (servf) {	fclose(servf);	servf = NULL;    }    serv_stayopen = 0;    UNLOCK;}libc_hidden_def(endservent)libc_hidden_proto(getservent_r)int getservent_r(struct servent * result_buf,		 char * buf, size_t buflen,		 struct servent ** result){    char *p;    register char *cp, **q;    char **serv_aliases;    char *line;    *result=NULL;    if (buflen < sizeof(*serv_aliases)*MAXALIASES) {	errno=ERANGE;	return errno;    }    LOCK;    serv_aliases=(char **)buf;    buf+=sizeof(*serv_aliases)*MAXALIASES;    buflen-=sizeof(*serv_aliases)*MAXALIASES;    if (buflen < BUFSIZ+1) {	UNLOCK;	errno=ERANGE;	return errno;    }    line=buf;    buf+=BUFSIZ+1;    buflen-=BUFSIZ+1;    if (servf == NULL && (servf = fopen(_PATH_SERVICES, "r" )) == NULL) {	UNLOCK;	errno=EIO;	return errno;    }again:    if ((p = fgets(line, BUFSIZ, servf)) == NULL) {	UNLOCK;	errno=EIO;	return errno;    }    if (*p == '#')	goto again;    cp = strpbrk(p, "#\n");    if (cp == NULL)	goto again;    *cp = '\0';    result_buf->s_name = p;    p = strpbrk(p, " \t");    if (p == NULL)	goto again;    *p++ = '\0';    while (*p == ' ' || *p == '\t')	p++;    cp = strpbrk(p, ",/");    if (cp == NULL)	goto again;    *cp++ = '\0';    result_buf->s_port = htons((u_short)atoi(p));    result_buf->s_proto = cp;    q = result_buf->s_aliases = serv_aliases;    cp = strpbrk(cp, " \t");    if (cp != NULL)	*cp++ = '\0';    while (cp && *cp) {	if (*cp == ' ' || *cp == '\t') {	    cp++;	    continue;	}	if (q < &serv_aliases[MAXALIASES - 1])	    *q++ = cp;	cp = strpbrk(cp, " \t");	if (cp != NULL)	    *cp++ = '\0';    }    *q = NULL;    *result=result_buf;    UNLOCK;    return 0;}libc_hidden_def(getservent_r)struct servent * getservent(void){    struct servent *result;    __initbuf();    getservent_r(&serv, servbuf, SBUFSIZE, &result);    return result;}libc_hidden_proto(getservbyname_r)int getservbyname_r(const char *name, const char *proto,	struct servent * result_buf, char * buf, size_t buflen,	struct servent ** result){    register char **cp;    int ret;    LOCK;    setservent(serv_stayopen);    while (!(ret=getservent_r(result_buf, buf, buflen, result))) {	if (strcmp(name, result_buf->s_name) == 0)	    goto gotname;	for (cp = result_buf->s_aliases; *cp; cp++)	    if (strcmp(name, *cp) == 0)		goto gotname;	continue;gotname:	if (proto == 0 || strcmp(result_buf->s_proto, proto) == 0)	    break;    }    if (!serv_stayopen)	endservent();    UNLOCK;    return *result?0:ret;}libc_hidden_def(getservbyname_r)struct servent *getservbyname(const char *name, const char *proto){    struct servent *result;    __initbuf();    getservbyname_r(name, proto, &serv, servbuf, SBUFSIZE, &result);    return result;}libc_hidden_proto(getservbyport_r)int getservbyport_r(int port, const char *proto,	struct servent * result_buf, char * buf,	size_t buflen, struct servent ** result){    int ret;    LOCK;    setservent(serv_stayopen);    while (!(ret=getservent_r(result_buf, buf, buflen, result))) {	if (result_buf->s_port != port)	    continue;	if (proto == 0 || strcmp(result_buf->s_proto, proto) == 0)	    break;    }    if (!serv_stayopen)	endservent();    UNLOCK;    return *result?0:ret;}libc_hidden_def(getservbyport_r)libc_hidden_proto(getservbyport)struct servent * getservbyport(int port, const char *proto){    struct servent *result;    __initbuf();    getservbyport_r(port, proto, &serv, servbuf, SBUFSIZE, &result);    return result;}libc_hidden_def(getservbyport)
 |