| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280 | /*** protocols.c                           /etc/protocols 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 <stdlib.h>#include <string.h>#include <errno.h>libc_hidden_proto(fopen)libc_hidden_proto(strcmp)libc_hidden_proto(strpbrk)libc_hidden_proto(atoi)libc_hidden_proto(rewind)libc_hidden_proto(fgets)libc_hidden_proto(fclose)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 *protof = NULL;static struct protoent proto;static char *static_aliases = NULL;static int proto_stayopen;static void __initbuf(void){    if (!static_aliases) {	static_aliases = malloc(SBUFSIZE);	if (!static_aliases)	    abort();    }}libc_hidden_proto(setprotoent)void setprotoent(int f){    LOCK;    if (protof == NULL)	protof = fopen(_PATH_PROTOCOLS, "r" );    else	rewind(protof);    proto_stayopen |= f;    UNLOCK;}libc_hidden_def(setprotoent)libc_hidden_proto(endprotoent)void endprotoent(void){    LOCK;    if (protof) {	fclose(protof);	protof = NULL;    }    proto_stayopen = 0;    UNLOCK;}libc_hidden_def(endprotoent)libc_hidden_proto(getprotoent_r)int getprotoent_r(struct protoent *result_buf,		  char *buf, size_t buflen,		  struct protoent **result){    char *p;    register char *cp, **q;    char **proto_aliases;    char *line;    *result = NULL;    if (buflen < sizeof(*proto_aliases)*MAXALIASES) {	errno=ERANGE;	return errno;    }    LOCK;    proto_aliases=(char **)buf;    buf+=sizeof(*proto_aliases)*MAXALIASES;    buflen-=sizeof(*proto_aliases)*MAXALIASES;    if (buflen < BUFSIZ+1) {	UNLOCK;	errno=ERANGE;	return errno;    }    line=buf;    buf+=BUFSIZ+1;    buflen-=BUFSIZ+1;    if (protof == NULL && (protof = fopen(_PATH_PROTOCOLS, "r" )) == NULL) {	UNLOCK;	return errno;    }again:    if ((p = fgets(line, BUFSIZ, protof)) == NULL) {	UNLOCK;	return TRY_AGAIN;    }    if (*p == '#')	goto again;    cp = strpbrk(p, "#\n");    if (cp == NULL)	goto again;    *cp = '\0';    result_buf->p_name = p;    cp = strpbrk(p, " \t");    if (cp == NULL)	goto again;    *cp++ = '\0';    while (*cp == ' ' || *cp == '\t')	cp++;    p = strpbrk(cp, " \t");    if (p != NULL)	*p++ = '\0';    result_buf->p_proto = atoi(cp);    q = result_buf->p_aliases = proto_aliases;    if (p != NULL) {	cp = p;	while (cp && *cp) {	    if (*cp == ' ' || *cp == '\t') {		cp++;		continue;	    }	    if (q < &proto_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(getprotoent_r)struct protoent * getprotoent(void){    struct protoent *result;    __initbuf();    getprotoent_r(&proto, static_aliases, SBUFSIZE, &result);    return result;}libc_hidden_proto(getprotobyname_r)int getprotobyname_r(const char *name,		    struct protoent *result_buf,		    char *buf, size_t buflen,		    struct protoent **result){    register char **cp;    int ret;    LOCK;    setprotoent(proto_stayopen);    while (!(ret=getprotoent_r(result_buf, buf, buflen, result))) {	if (strcmp(result_buf->p_name, name) == 0)	    break;	for (cp = result_buf->p_aliases; *cp != 0; cp++)	    if (strcmp(*cp, name) == 0)		goto found;    }found:    if (!proto_stayopen)	endprotoent();    UNLOCK;    return *result?0:ret;}libc_hidden_def(getprotobyname_r)struct protoent * getprotobyname(const char *name){    struct protoent *result;    __initbuf();    getprotobyname_r(name, &proto, static_aliases, SBUFSIZE, &result);    return result;}libc_hidden_proto(getprotobynumber_r)int getprotobynumber_r (int proto_num,			struct protoent *result_buf,			char *buf, size_t buflen,			struct protoent **result){    int ret;    LOCK;    setprotoent(proto_stayopen);    while (!(ret=getprotoent_r(result_buf, buf, buflen, result)))	if (result_buf->p_proto == proto_num)	    break;    if (!proto_stayopen)	endprotoent();    UNLOCK;    return *result?0:ret;}libc_hidden_def(getprotobynumber_r)struct protoent * getprotobynumber(int proto_num){    struct protoent *result;    __initbuf();    getprotobynumber_r(proto_num, &proto, static_aliases,                       SBUFSIZE, &result);    return result;}
 |