/* @(#)getrpcent.c 2.2 88/07/29 4.0 RPCSRC */ /* * Sun RPC is a product of Sun Microsystems, Inc. and is provided for * unrestricted use provided that this legend is included on all tape * media and as a part of the software program in whole or part. Users * may copy or modify Sun RPC without charge, but are not authorized * to license or distribute it to anyone else except as part of a product or * program developed by the user. * * SUN RPC IS PROVIDED AS IS WITH NO WARRANTIES OF ANY KIND INCLUDING THE * WARRANTIES OF DESIGN, MERCHANTIBILITY AND FITNESS FOR A PARTICULAR * PURPOSE, OR ARISING FROM A COURSE OF DEALING, USAGE OR TRADE PRACTICE. * * Sun RPC is provided with no support and without any obligation on the * part of Sun Microsystems, Inc. to assist in its use, correction, * modification or enhancement. * * SUN MICROSYSTEMS, INC. SHALL HAVE NO LIABILITY WITH RESPECT TO THE * INFRINGEMENT OF COPYRIGHTS, TRADE SECRETS OR ANY PATENTS BY SUN RPC * OR ANY PART THEREOF. * * In no event will Sun Microsystems, Inc. be liable for any lost revenue * or profits or other special, indirect and consequential damages, even if * Sun has been advised of the possibility of such damages. * * Sun Microsystems, Inc. * 2550 Garcia Avenue * Mountain View, California 94043 */ /* * Copyright (c) 1985 by Sun Microsystems, Inc. */ #define atoi __atoi #define rewind __rewind #define __FORCE_GLIBC #include #include #include #include #include #include #include #include #include /* * Internet version. */ static struct rpcdata { FILE *rpcf; char *current; int currentlen; int stayopen; #define MAXALIASES 35 char *rpc_aliases[MAXALIASES]; struct rpcent rpc; char line[BUFSIZ + 1]; char *domain; } *rpcdata; static char RPCDB[] = "/etc/rpc"; static struct rpcdata *_rpcdata(void) { register struct rpcdata *d = rpcdata; if (d == NULL) { d = (struct rpcdata *) calloc(1, sizeof(struct rpcdata)); rpcdata = d; } return d; } void attribute_hidden __endrpcent(void) { register struct rpcdata *d = _rpcdata(); if (d == NULL) return; if (d->stayopen) return; if (d->current) { free(d->current); d->current = NULL; } if (d->rpcf) { fclose(d->rpcf); d->rpcf = NULL; } } strong_alias(__endrpcent,endrpcent) void attribute_hidden __setrpcent(int f) { register struct rpcdata *d = _rpcdata(); if (d == NULL) return; if (d->rpcf == NULL) d->rpcf = fopen(RPCDB, "r"); else rewind(d->rpcf); if (d->current) free(d->current); d->current = NULL; d->stayopen |= f; } strong_alias(__setrpcent,setrpcent) static struct rpcent *interpret(struct rpcdata *); static struct rpcent *__get_next_rpcent(struct rpcdata *d) { if (fgets(d->line, BUFSIZ, d->rpcf) == NULL) return NULL; return interpret(d); } struct rpcent attribute_hidden *__getrpcent(void) { register struct rpcdata *d = _rpcdata(); if (d == NULL) return NULL; if (d->rpcf == NULL && (d->rpcf = fopen(RPCDB, "r")) == NULL) return NULL; return __get_next_rpcent(d); } strong_alias(__getrpcent,getrpcent) struct rpcent attribute_hidden *__getrpcbynumber(register int number) { register struct rpcdata *d = _rpcdata(); register struct rpcent *rpc; if (d == NULL) return NULL; __setrpcent(0); while ((rpc = __getrpcent())) { if (rpc->r_number == number) break; } __endrpcent(); return rpc; } strong_alias(__getrpcbynumber,getrpcbynumber) struct rpcent attribute_hidden *__getrpcbyname(const char *name) { struct rpcent *rpc; char **rp; __setrpcent(0); while ((rpc = __getrpcent())) { if (__strcmp(rpc->r_name, name) == 0) return rpc; for (rp = rpc->r_aliases; *rp != NULL; rp++) { if (__strcmp(*rp, name) == 0) return rpc; } } __endrpcent(); return NULL; } strong_alias(__getrpcbyname,getrpcbyname) #ifdef __linux__ static char *firstwhite(char *s) { char *s1, *s2; s1 = __strchr(s, ' '); s2 = __strchr(s, '\t'); if (s1) { if (s2) return (s1 < s2) ? s1 : s2; else return s1; } else return s2; } #endif static struct rpcent *interpret(register struct rpcdata *d) { char *p; register char *cp, **q; p = d->line; d->line[__strlen(p)-1] = '\n'; if (*p == '#') return __get_next_rpcent(d); cp = __strchr(p, '#'); if (cp == NULL) { cp = __strchr(p, '\n'); if (cp == NULL) return __get_next_rpcent(d); } *cp = '\0'; #ifdef __linux__ if ((cp = firstwhite(p))) *cp++ = 0; else return __get_next_rpcent(d); #else cp = __strchr(p, ' '); if (cp == NULL) { cp = __strchr(p, '\t'); if (cp == NULL) return __get_next_rpcent(d); } *cp++ = '\0'; #endif /* THIS STUFF IS INTERNET SPECIFIC */ d->rpc.r_name = d->line; while (*cp == ' ' || *cp == '\t') cp++; d->rpc.r_number = atoi(cp); q = d->rpc.r_aliases = d->rpc_aliases; #ifdef __linux__ if ((cp = firstwhite(cp))) *cp++ = '\0'; #else cp = __strchr(p, ' '); if (cp != NULL) *cp++ = '\0'; else { cp = __strchr(p, '\t'); if (cp != NULL) *cp++ = '\0'; } #endif while (cp && *cp) { if (*cp == ' ' || *cp == '\t') { cp++; continue; } if (q < &(d->rpc_aliases[MAXALIASES - 1])) *q++ = cp; #ifdef __linux__ if ((cp = firstwhite(cp))) *cp++ = '\0'; #else cp = __strchr(p, ' '); if (cp != NULL) *cp++ = '\0'; else { cp = __strchr(p, '\t'); if (cp != NULL) *cp++ = '\0'; } #endif } *q = NULL; return &d->rpc; } #if defined(__UCLIBC_HAS_REENTRANT_RPC__) #if defined(__UCLIBC_HAS_THREADS__) # include static pthread_mutex_t rpcdata_lock = PTHREAD_MUTEX_INITIALIZER; #endif #define LOCK __pthread_mutex_lock(&rpcdata_lock) #define UNLOCK __pthread_mutex_unlock(&rpcdata_lock) static int __copy_rpcent(struct rpcent *r, struct rpcent *result_buf, char *buffer, size_t buflen, struct rpcent **result) { size_t i, s; *result = NULL; if (!r) return ENOENT; /* copy the struct from the shared mem */ __memset(result_buf, 0x00, sizeof(*result_buf)); __memset(buffer, 0x00, buflen); result_buf->r_number = r->r_number; /* copy the aliases ... need to not only copy the alias strings, * but the array of pointers to the alias strings */ i = 0; while (r->r_aliases[i++]) ; s = i-- * sizeof(char*); if (buflen < s) goto err_out; result_buf->r_aliases = (char**)buffer; buffer += s; buflen -= s; while (i-- > 0) { s = __strlen(r->r_aliases[i]) + 1; if (buflen < s) goto err_out; result_buf->r_aliases[i] = buffer; buffer += s; buflen -= s; __memcpy(result_buf->r_aliases[i], r->r_aliases[i], s); } /* copy the name */ i = __strlen(r->r_name); if (buflen <= i) goto err_out; result_buf->r_name = buffer; __memcpy(result_buf->r_name, r->r_name, i); /* that was a hoot eh ? */ *result = result_buf; return 0; err_out: return ERANGE; } int getrpcbynumber_r(int number, struct rpcent *result_buf, char *buffer, size_t buflen, struct rpcent **result) { int ret; LOCK; ret = __copy_rpcent(__getrpcbynumber(number), result_buf, buffer, buflen, result); UNLOCK; return ret; } int getrpcbyname_r(const char *name, struct rpcent *result_buf, char *buffer, size_t buflen, struct rpcent **result) { int ret; LOCK; ret = __copy_rpcent(__getrpcbyname(name), result_buf, buffer, buflen, result); UNLOCK; return ret; } int getrpcent_r(struct rpcent *result_buf, char *buffer, size_t buflen, struct rpcent **result) { int ret; LOCK; ret = __copy_rpcent(__getrpcent(), result_buf, buffer, buflen, result); UNLOCK; return ret; } #endif /* __UCLIBC_HAS_REENTRANT_RPC__ */