xref: /freebsd/lib/libc/rpc/getrpcent.c (revision 646a7fea0c8a60ce2795ffc1bdf58e0fd0f7d624)
1 /*	$NetBSD: getrpcent.c,v 1.17 2000/01/22 22:19:17 mycroft Exp $	*/
2 
3 /*
4  * Sun RPC is a product of Sun Microsystems, Inc. and is provided for
5  * unrestricted use provided that this legend is included on all tape
6  * media and as a part of the software program in whole or part.  Users
7  * may copy or modify Sun RPC without charge, but are not authorized
8  * to license or distribute it to anyone else except as part of a product or
9  * program developed by the user or with the express written consent of
10  * Sun Microsystems, Inc.
11  *
12  * SUN RPC IS PROVIDED AS IS WITH NO WARRANTIES OF ANY KIND INCLUDING THE
13  * WARRANTIES OF DESIGN, MERCHANTIBILITY AND FITNESS FOR A PARTICULAR
14  * PURPOSE, OR ARISING FROM A COURSE OF DEALING, USAGE OR TRADE PRACTICE.
15  *
16  * Sun RPC is provided with no support and without any obligation on the
17  * part of Sun Microsystems, Inc. to assist in its use, correction,
18  * modification or enhancement.
19  *
20  * SUN MICROSYSTEMS, INC. SHALL HAVE NO LIABILITY WITH RESPECT TO THE
21  * INFRINGEMENT OF COPYRIGHTS, TRADE SECRETS OR ANY PATENTS BY SUN RPC
22  * OR ANY PART THEREOF.
23  *
24  * In no event will Sun Microsystems, Inc. be liable for any lost revenue
25  * or profits or other special, indirect and consequential damages, even if
26  * Sun has been advised of the possibility of such damages.
27  *
28  * Sun Microsystems, Inc.
29  * 2550 Garcia Avenue
30  * Mountain View, California  94043
31  */
32 
33 #if defined(LIBC_SCCS) && !defined(lint)
34 static char *sccsid = "@(#)getrpcent.c 1.14 91/03/11 Copyr 1984 Sun Micro";
35 #endif
36 #include <sys/cdefs.h>
37 __FBSDID("$FreeBSD$");
38 
39 /*
40  * Copyright (c) 1984 by Sun Microsystems, Inc.
41  */
42 
43 #include <sys/param.h>
44 #include <sys/types.h>
45 #include <sys/socket.h>
46 #include <arpa/inet.h>
47 #include <assert.h>
48 #include <errno.h>
49 #include <nsswitch.h>
50 #include <netinet/in.h>
51 #include <stdio.h>
52 #include <string.h>
53 #include <stdarg.h>
54 #include <stdlib.h>
55 #include <rpc/rpc.h>
56 #ifdef YP
57 #include <rpcsvc/yp_prot.h>
58 #include <rpcsvc/ypclnt.h>
59 #endif
60 #include <unistd.h>
61 #include "namespace.h"
62 #include "reentrant.h"
63 #include "un-namespace.h"
64 #include "libc_private.h"
65 #include "nss_tls.h"
66 #ifdef NS_CACHING
67 #include "nscache.h"
68 #endif
69 
70 #define	RPCDB	"/etc/rpc"
71 
72 /* nsswitch declarations */
73 enum constants
74 {
75 	SETRPCENT = 1,
76 	ENDRPCENT = 2,
77 	RPCENT_STORAGE_INITIAL	= 1 << 10, /* 1 KByte */
78 	RPCENT_STORAGE_MAX	= 1 << 20, /* 1 MByte */
79 };
80 
81 static const ns_src defaultsrc[] = {
82 	{ NSSRC_FILES, NS_SUCCESS },
83 #ifdef YP
84 	{ NSSRC_NIS, NS_SUCCESS },
85 #endif
86 	{ NULL, 0 }
87 };
88 
89 /* files backend declarations */
90 struct files_state {
91 	FILE	*fp;
92 	int	stayopen;
93 };
94 
95 static	int	files_rpcent(void *, void *, va_list);
96 static	int	files_setrpcent(void *, void *, va_list);
97 
98 static	void	files_endstate(void *);
99 NSS_TLS_HANDLING(files);
100 
101 /* nis backend declarations */
102 #ifdef YP
103 struct nis_state {
104 	char	domain[MAXHOSTNAMELEN];
105 	char	*current;
106 	int	currentlen;
107 	int	stepping;
108 	int	no_name_map;
109 };
110 
111 static	int	nis_rpcent(void *, void *, va_list);
112 static	int	nis_setrpcent(void *, void *, va_list);
113 
114 static	void	nis_endstate(void *);
115 NSS_TLS_HANDLING(nis);
116 #endif
117 
118 /* get** wrappers for get**_r functions declarations */
119 struct rpcent_state {
120 	struct rpcent	rpc;
121 	char		*buffer;
122 	size_t	bufsize;
123 };
124 static	void	rpcent_endstate(void *);
125 NSS_TLS_HANDLING(rpcent);
126 
127 union key {
128 	const char	*name;
129 	int		number;
130 };
131 
132 static int wrap_getrpcbyname_r(union key, struct rpcent *, char *,
133 			size_t, struct rpcent **);
134 static int wrap_getrpcbynumber_r(union key, struct rpcent *, char *,
135 			size_t, struct rpcent **);
136 static int wrap_getrpcent_r(union key, struct rpcent *, char *,
137 			size_t, struct rpcent **);
138 static struct rpcent *getrpc(int (*fn)(union key, struct rpcent *, char *,
139 			size_t, struct rpcent **), union key);
140 
141 #ifdef NS_CACHING
142 static int rpc_id_func(char *, size_t *, va_list, void *);
143 static int rpc_marshal_func(char *, size_t *, void *, va_list, void *);
144 static int rpc_unmarshal_func(char *, size_t, void *, va_list, void *);
145 #endif
146 
147 static int
148 rpcent_unpack(char *p, struct rpcent *rpc, char **r_aliases,
149 	size_t aliases_size, int *errnop)
150 {
151 	char *cp, **q;
152 
153 	assert(p != NULL);
154 
155 	if (*p == '#')
156 		return (-1);
157 	cp = strpbrk(p, "#\n");
158 	if (cp == NULL)
159 		return (-1);
160 	*cp = '\0';
161 	cp = strpbrk(p, " \t");
162 	if (cp == NULL)
163 		return (-1);
164 	*cp++ = '\0';
165 	/* THIS STUFF IS INTERNET SPECIFIC */
166 	rpc->r_name = p;
167 	while (*cp == ' ' || *cp == '\t')
168 		cp++;
169 	rpc->r_number = atoi(cp);
170 	q = rpc->r_aliases = r_aliases;
171 	cp = strpbrk(cp, " \t");
172 	if (cp != NULL)
173 		*cp++ = '\0';
174 	while (cp && *cp) {
175 		if (*cp == ' ' || *cp == '\t') {
176 			cp++;
177 			continue;
178 		}
179 		if (q < &(r_aliases[aliases_size - 1]))
180 			*q++ = cp;
181 		else {
182 			*errnop = ERANGE;
183 			return -1;
184 		}
185 
186 		cp = strpbrk(cp, " \t");
187 		if (cp != NULL)
188 			*cp++ = '\0';
189 	}
190 	*q = NULL;
191 	return 0;
192 }
193 
194 /* files backend implementation */
195 static	void
196 files_endstate(void *p)
197 {
198 	FILE * f;
199 
200 	if (p == NULL)
201 		return;
202 
203 	f = ((struct files_state *)p)->fp;
204 	if (f != NULL)
205 		fclose(f);
206 
207 	free(p);
208 }
209 
210 static int
211 files_rpcent(void *retval, void *mdata, va_list ap)
212 {
213 	char *name;
214 	int number;
215 	struct rpcent *rpc;
216 	char *buffer;
217 	size_t bufsize;
218 	int *errnop;
219 
220 	char *line;
221 	size_t linesize;
222 	char **aliases;
223 	int aliases_size;
224 	char **rp;
225 
226 	struct files_state	*st;
227 	int rv;
228 	int stayopen;
229 	enum nss_lookup_type how;
230 
231 	how = (enum nss_lookup_type)mdata;
232 	switch (how)
233 	{
234 	case nss_lt_name:
235 		name = va_arg(ap, char *);
236 		break;
237 	case nss_lt_id:
238 		number = va_arg(ap, int);
239 		break;
240 	case nss_lt_all:
241 		break;
242 	default:
243 		return (NS_NOTFOUND);
244 	}
245 
246 	rpc = va_arg(ap, struct rpcent *);
247 	buffer = va_arg(ap, char *);
248 	bufsize = va_arg(ap, size_t);
249 	errnop = va_arg(ap, int *);
250 
251 	*errnop = files_getstate(&st);
252 	if (*errnop != 0)
253 		return (NS_UNAVAIL);
254 
255 	if (st->fp == NULL && (st->fp = fopen(RPCDB, "r")) == NULL) {
256 		*errnop = errno;
257 		return (NS_UNAVAIL);
258 	}
259 
260 	if (how == nss_lt_all)
261 		stayopen = 1;
262 	else {
263 		rewind(st->fp);
264 		stayopen = st->stayopen;
265 	}
266 
267 	do {
268 		if ((line = fgetln(st->fp, &linesize)) == NULL) {
269 			*errnop = errno;
270 			rv = NS_RETURN;
271 			break;
272 		}
273 
274 		if (bufsize <= linesize + _ALIGNBYTES + sizeof(char *)) {
275 			*errnop = ERANGE;
276 			rv = NS_RETURN;
277 			break;
278 		}
279 
280 		aliases = (char **)_ALIGN(&buffer[linesize+1]);
281 		aliases_size = (buffer + bufsize -
282 			(char *)aliases)/sizeof(char *);
283 		if (aliases_size < 1) {
284 			*errnop = ERANGE;
285 			rv = NS_RETURN;
286 			break;
287 		}
288 
289 		memcpy(buffer, line, linesize);
290 		buffer[linesize] = '\0';
291 
292 		rv = rpcent_unpack(buffer, rpc, aliases, aliases_size, errnop);
293 		if (rv != 0) {
294 			if (*errnop == 0) {
295 				rv = NS_NOTFOUND;
296 				continue;
297 			}
298 			else {
299 				rv = NS_RETURN;
300 				break;
301 			}
302 		}
303 
304 		switch (how)
305 		{
306 		case nss_lt_name:
307 			if (strcmp(rpc->r_name, name) == 0)
308 				goto done;
309 			for (rp = rpc->r_aliases; *rp != NULL; rp++) {
310 				if (strcmp(*rp, name) == 0)
311 					goto done;
312 			}
313 			rv = NS_NOTFOUND;
314 			continue;
315 done:
316 			rv = NS_SUCCESS;
317 			break;
318 		case nss_lt_id:
319 			rv = (rpc->r_number == number) ? NS_SUCCESS :
320 				NS_NOTFOUND;
321 			break;
322 		case nss_lt_all:
323 			rv = NS_SUCCESS;
324 			break;
325 		}
326 
327 	} while (!(rv & NS_TERMINATE));
328 
329 	if (!stayopen && st->fp!=NULL) {
330 		fclose(st->fp);
331 		st->fp = NULL;
332 	}
333 
334 	if ((rv == NS_SUCCESS) && (retval != NULL))
335 		*((struct rpcent **)retval) = rpc;
336 
337 	return (rv);
338 }
339 
340 static int
341 files_setrpcent(void *retval, void *mdata, va_list ap)
342 {
343 	struct files_state	*st;
344 	int	rv;
345 	int	f;
346 
347 	rv = files_getstate(&st);
348 	if (rv != 0)
349 		return (NS_UNAVAIL);
350 
351 	switch ((enum constants)mdata)
352 	{
353 	case SETRPCENT:
354 		f = va_arg(ap,int);
355 		if (st->fp == NULL)
356 			st->fp = fopen(RPCDB, "r");
357 		else
358 			rewind(st->fp);
359 		st->stayopen |= f;
360 		break;
361 	case ENDRPCENT:
362 		if (st->fp != NULL) {
363 			fclose(st->fp);
364 			st->fp = NULL;
365 		}
366 		st->stayopen = 0;
367 		break;
368 	default:
369 		break;
370 	}
371 
372 	return (NS_UNAVAIL);
373 }
374 
375 /* nis backend implementation */
376 #ifdef YP
377 static 	void
378 nis_endstate(void *p)
379 {
380 	if (p == NULL)
381 		return;
382 
383 	free(((struct nis_state *)p)->current);
384 	free(p);
385 }
386 
387 static int
388 nis_rpcent(void *retval, void *mdata, va_list ap)
389 {
390 	char		*name;
391 	int		number;
392 	struct rpcent	*rpc;
393 	char		*buffer;
394 	size_t	bufsize;
395 	int		*errnop;
396 
397 	char		**rp;
398 	char		**aliases;
399 	int		aliases_size;
400 
401 	char	*lastkey;
402 	char	*resultbuf;
403 	int	resultbuflen;
404 	char	buf[YPMAXRECORD + 2];
405 
406 	struct nis_state	*st;
407 	int		rv;
408 	enum nss_lookup_type	how;
409 	int	no_name_active;
410 
411 	how = (enum nss_lookup_type)mdata;
412 	switch (how)
413 	{
414 	case nss_lt_name:
415 		name = va_arg(ap, char *);
416 		break;
417 	case nss_lt_id:
418 		number = va_arg(ap, int);
419 		break;
420 	case nss_lt_all:
421 		break;
422 	default:
423 		return (NS_NOTFOUND);
424 	}
425 
426 	rpc = va_arg(ap, struct rpcent *);
427 	buffer = va_arg(ap, char *);
428 	bufsize = va_arg(ap, size_t);
429 	errnop = va_arg(ap, int *);
430 
431 	*errnop = nis_getstate(&st);
432 	if (*errnop != 0)
433 		return (NS_UNAVAIL);
434 
435 	if (st->domain[0] == '\0') {
436 		if (getdomainname(st->domain, sizeof(st->domain)) != 0) {
437 			*errnop = errno;
438 			return (NS_UNAVAIL);
439 		}
440 	}
441 
442 	no_name_active = 0;
443 	do {
444 		switch (how)
445 		{
446 		case nss_lt_name:
447 			if (!st->no_name_map)
448 			{
449 				snprintf(buf, sizeof buf, "%s", name);
450 				rv = yp_match(st->domain, "rpc.byname", buf,
451 			    		strlen(buf), &resultbuf, &resultbuflen);
452 
453 				switch (rv) {
454 				case 0:
455 					break;
456 				case YPERR_MAP:
457 					st->stepping = 0;
458 					no_name_active = 1;
459 					how = nss_lt_all;
460 
461 					rv = NS_NOTFOUND;
462 					continue;
463 				default:
464 					rv = NS_NOTFOUND;
465 					goto fin;
466 				}
467 			} else {
468 				st->stepping = 0;
469 				no_name_active = 1;
470 				how = nss_lt_all;
471 
472 				rv = NS_NOTFOUND;
473 				continue;
474 			}
475 		break;
476 		case nss_lt_id:
477 			snprintf(buf, sizeof buf, "%d", number);
478 			if (yp_match(st->domain, "rpc.bynumber", buf,
479 			    	strlen(buf), &resultbuf, &resultbuflen)) {
480 				rv = NS_NOTFOUND;
481 				goto fin;
482 			}
483 			break;
484 		case nss_lt_all:
485 				if (!st->stepping) {
486 					rv = yp_first(st->domain, "rpc.bynumber",
487 				    		&st->current,
488 						&st->currentlen, &resultbuf,
489 				    		&resultbuflen);
490 					if (rv) {
491 						rv = NS_NOTFOUND;
492 						goto fin;
493 					}
494 					st->stepping = 1;
495 				} else {
496 					lastkey = st->current;
497 					rv = yp_next(st->domain, "rpc.bynumber",
498 				    		st->current,
499 						st->currentlen, &st->current,
500 				    		&st->currentlen,
501 						&resultbuf,	&resultbuflen);
502 					free(lastkey);
503 					if (rv) {
504 						st->stepping = 0;
505 						rv = NS_NOTFOUND;
506 						goto fin;
507 					}
508 				}
509 			break;
510 		}
511 
512 		/* we need a room for additional \n symbol */
513 		if (bufsize <= resultbuflen + 1 + _ALIGNBYTES +
514 		    sizeof(char *)) {
515 			*errnop = ERANGE;
516 			rv = NS_RETURN;
517 			break;
518 		}
519 
520 		aliases=(char **)_ALIGN(&buffer[resultbuflen+2]);
521 		aliases_size = (buffer + bufsize - (char *)aliases) /
522 			sizeof(char *);
523 		if (aliases_size < 1) {
524 			*errnop = ERANGE;
525 			rv = NS_RETURN;
526 			break;
527 		}
528 
529 		/*
530 		 * rpcent_unpack expects lines terminated with \n -- make it happy
531 		 */
532 		memcpy(buffer, resultbuf, resultbuflen);
533 		buffer[resultbuflen] = '\n';
534 		buffer[resultbuflen+1] = '\0';
535 		free(resultbuf);
536 
537 		if (rpcent_unpack(buffer, rpc, aliases, aliases_size,
538 		    errnop) != 0) {
539 			if (*errnop == 0)
540 				rv = NS_NOTFOUND;
541 			else
542 				rv = NS_RETURN;
543 		} else {
544 			if ((how == nss_lt_all) && (no_name_active != 0)) {
545 				if (strcmp(rpc->r_name, name) == 0)
546 					goto done;
547 				for (rp = rpc->r_aliases; *rp != NULL; rp++) {
548 					if (strcmp(*rp, name) == 0)
549 						goto done;
550 				}
551 				rv = NS_NOTFOUND;
552 				continue;
553 done:
554 				rv = NS_SUCCESS;
555 			} else
556 				rv = NS_SUCCESS;
557 		}
558 
559 	} while (!(rv & NS_TERMINATE) && (how == nss_lt_all));
560 
561 fin:
562 	if ((rv == NS_SUCCESS) && (retval != NULL))
563 		*((struct rpcent **)retval) = rpc;
564 
565 	return (rv);
566 }
567 
568 static int
569 nis_setrpcent(void *retval, void *mdata, va_list ap)
570 {
571 	struct nis_state	*st;
572 	int	rv;
573 
574 	rv = nis_getstate(&st);
575 	if (rv != 0)
576 		return (NS_UNAVAIL);
577 
578 	switch ((enum constants)mdata)
579 	{
580 	case SETRPCENT:
581 	case ENDRPCENT:
582 		free(st->current);
583 		st->current = NULL;
584 		st->stepping = 0;
585 		break;
586 	default:
587 		break;
588 	}
589 
590 	return (NS_UNAVAIL);
591 }
592 #endif
593 
594 #ifdef NS_CACHING
595 static int
596 rpc_id_func(char *buffer, size_t *buffer_size, va_list ap, void *cache_mdata)
597 {
598 	char *name;
599 	int rpc;
600 
601 	size_t desired_size, size;
602 	enum nss_lookup_type lookup_type;
603 	int res = NS_UNAVAIL;
604 
605 	lookup_type = (enum nss_lookup_type)cache_mdata;
606 	switch (lookup_type) {
607 	case nss_lt_name:
608 		name = va_arg(ap, char *);
609 
610 		size = strlen(name);
611 		desired_size = sizeof(enum nss_lookup_type) + size + 1;
612 		if (desired_size > *buffer_size) {
613 			res = NS_RETURN;
614 			goto fin;
615 		}
616 
617 		memcpy(buffer, &lookup_type, sizeof(enum nss_lookup_type));
618 		memcpy(buffer + sizeof(enum nss_lookup_type), name, size + 1);
619 
620 		res = NS_SUCCESS;
621 		break;
622 	case nss_lt_id:
623 		rpc = va_arg(ap, int);
624 
625 		desired_size = sizeof(enum nss_lookup_type) + sizeof(int);
626 		if (desired_size > *buffer_size) {
627 			res = NS_RETURN;
628 			goto fin;
629 		}
630 
631 		memcpy(buffer, &lookup_type, sizeof(enum nss_lookup_type));
632 		memcpy(buffer + sizeof(enum nss_lookup_type), &rpc,
633 		    sizeof(int));
634 
635 		res = NS_SUCCESS;
636 		break;
637 	default:
638 		/* should be unreachable */
639 		return (NS_UNAVAIL);
640 	}
641 
642 fin:
643 	*buffer_size = desired_size;
644 	return (res);
645 }
646 
647 static int
648 rpc_marshal_func(char *buffer, size_t *buffer_size, void *retval, va_list ap,
649     void *cache_mdata)
650 {
651 	char *name;
652 	int num;
653 	struct rpcent *rpc;
654 	char *orig_buf;
655 	size_t orig_buf_size;
656 
657 	struct rpcent new_rpc;
658 	size_t desired_size, size, aliases_size;
659 	char *p;
660 	char **alias;
661 
662 	switch ((enum nss_lookup_type)cache_mdata) {
663 	case nss_lt_name:
664 		name = va_arg(ap, char *);
665 		break;
666 	case nss_lt_id:
667 		num = va_arg(ap, int);
668 		break;
669 	case nss_lt_all:
670 		break;
671 	default:
672 		/* should be unreachable */
673 		return (NS_UNAVAIL);
674 	}
675 
676 	rpc = va_arg(ap, struct rpcent *);
677 	orig_buf = va_arg(ap, char *);
678 	orig_buf_size = va_arg(ap, size_t);
679 
680 	desired_size = _ALIGNBYTES + sizeof(struct rpcent) + sizeof(char *);
681 	if (rpc->r_name != NULL)
682 		desired_size += strlen(rpc->r_name) + 1;
683 
684 	if (rpc->r_aliases != NULL) {
685 		aliases_size = 0;
686 		for (alias = rpc->r_aliases; *alias; ++alias) {
687 			desired_size += strlen(*alias) + 1;
688 			++aliases_size;
689 		}
690 
691 		desired_size += _ALIGNBYTES + (aliases_size + 1) *
692 		    sizeof(char *);
693 	}
694 
695 	if (*buffer_size < desired_size) {
696 		/* this assignment is here for future use */
697 		*buffer_size = desired_size;
698 		return (NS_RETURN);
699 	}
700 
701 	new_rpc = *rpc;
702 
703 	*buffer_size = desired_size;
704 	memset(buffer, 0, desired_size);
705 	p = buffer + sizeof(struct rpcent) + sizeof(char *);
706 	memcpy(buffer + sizeof(struct rpcent), &p, sizeof(char *));
707 	p = (char *)_ALIGN(p);
708 
709 	if (new_rpc.r_name != NULL) {
710 		size = strlen(new_rpc.r_name);
711 		memcpy(p, new_rpc.r_name, size);
712 		new_rpc.r_name = p;
713 		p += size + 1;
714 	}
715 
716 	if (new_rpc.r_aliases != NULL) {
717 		p = (char *)_ALIGN(p);
718 		memcpy(p, new_rpc.r_aliases, sizeof(char *) * aliases_size);
719 		new_rpc.r_aliases = (char **)p;
720 		p += sizeof(char *) * (aliases_size + 1);
721 
722 		for (alias = new_rpc.r_aliases; *alias; ++alias) {
723 			size = strlen(*alias);
724 			memcpy(p, *alias, size);
725 			*alias = p;
726 			p += size + 1;
727 		}
728 	}
729 
730 	memcpy(buffer, &new_rpc, sizeof(struct rpcent));
731 	return (NS_SUCCESS);
732 }
733 
734 static int
735 rpc_unmarshal_func(char *buffer, size_t buffer_size, void *retval, va_list ap,
736     void *cache_mdata)
737 {
738 	char *name;
739 	int num;
740 	struct rpcent *rpc;
741 	char *orig_buf;
742 	size_t orig_buf_size;
743 	int *ret_errno;
744 
745 	char *p;
746 	char **alias;
747 
748 	switch ((enum nss_lookup_type)cache_mdata) {
749 	case nss_lt_name:
750 		name = va_arg(ap, char *);
751 		break;
752 	case nss_lt_id:
753 		num = va_arg(ap, int);
754 		break;
755 	case nss_lt_all:
756 		break;
757 	default:
758 		/* should be unreachable */
759 		return (NS_UNAVAIL);
760 	}
761 
762 	rpc = va_arg(ap, struct rpcent *);
763 	orig_buf = va_arg(ap, char *);
764 	orig_buf_size = va_arg(ap, size_t);
765 	ret_errno = va_arg(ap, int *);
766 
767 	if (orig_buf_size <
768 	    buffer_size - sizeof(struct rpcent) - sizeof(char *)) {
769 		*ret_errno = ERANGE;
770 		return (NS_RETURN);
771 	}
772 
773 	memcpy(rpc, buffer, sizeof(struct rpcent));
774 	memcpy(&p, buffer + sizeof(struct rpcent), sizeof(char *));
775 
776 	orig_buf = (char *)_ALIGN(orig_buf);
777 	memcpy(orig_buf, buffer + sizeof(struct rpcent) + sizeof(char *) +
778 	    _ALIGN(p) - (size_t)p,
779 	    buffer_size - sizeof(struct rpcent) - sizeof(char *) -
780 	    _ALIGN(p) + (size_t)p);
781 	p = (char *)_ALIGN(p);
782 
783 	NS_APPLY_OFFSET(rpc->r_name, orig_buf, p, char *);
784 	if (rpc->r_aliases != NULL) {
785 		NS_APPLY_OFFSET(rpc->r_aliases, orig_buf, p, char **);
786 
787 		for (alias = rpc->r_aliases	; *alias; ++alias)
788 			NS_APPLY_OFFSET(*alias, orig_buf, p, char *);
789 	}
790 
791 	if (retval != NULL)
792 		*((struct rpcent **)retval) = rpc;
793 
794 	return (NS_SUCCESS);
795 }
796 
797 NSS_MP_CACHE_HANDLING(rpc);
798 #endif /* NS_CACHING */
799 
800 
801 /* get**_r functions implementation */
802 static int
803 getrpcbyname_r(const char *name, struct rpcent *rpc, char *buffer,
804 	size_t bufsize, struct rpcent **result)
805 {
806 #ifdef NS_CACHING
807 	static const nss_cache_info cache_info =
808     		NS_COMMON_CACHE_INFO_INITIALIZER(
809 		rpc, (void *)nss_lt_name,
810 		rpc_id_func, rpc_marshal_func, rpc_unmarshal_func);
811 #endif
812 	static const ns_dtab dtab[] = {
813 		{ NSSRC_FILES, files_rpcent, (void *)nss_lt_name },
814 #ifdef YP
815 		{ NSSRC_NIS, nis_rpcent, (void *)nss_lt_name },
816 #endif
817 #ifdef NS_CACHING
818 		NS_CACHE_CB(&cache_info)
819 #endif
820 		{ NULL, NULL, NULL }
821 	};
822 	int rv, ret_errno;
823 
824 	ret_errno = 0;
825 	*result = NULL;
826 	rv = nsdispatch(result, dtab, NSDB_RPC, "getrpcbyname_r", defaultsrc,
827 	    name, rpc, buffer, bufsize, &ret_errno);
828 
829 	if (rv == NS_SUCCESS)
830 		return (0);
831 	else
832 		return (ret_errno);
833 }
834 
835 static int
836 getrpcbynumber_r(int number, struct rpcent *rpc, char *buffer,
837 	size_t bufsize, struct rpcent **result)
838 {
839 #ifdef NS_CACHING
840 	static const nss_cache_info cache_info =
841     		NS_COMMON_CACHE_INFO_INITIALIZER(
842 		rpc, (void *)nss_lt_id,
843 		rpc_id_func, rpc_marshal_func, rpc_unmarshal_func);
844 #endif
845 	static const ns_dtab dtab[] = {
846 		{ NSSRC_FILES, files_rpcent, (void *)nss_lt_id },
847 #ifdef YP
848 		{ NSSRC_NIS, nis_rpcent, (void *)nss_lt_id },
849 #endif
850 #ifdef NS_CACHING
851 		NS_CACHE_CB(&cache_info)
852 #endif
853 		{ NULL, NULL, NULL }
854 	};
855 	int rv, ret_errno;
856 
857 	ret_errno = 0;
858 	*result = NULL;
859 	rv = nsdispatch(result, dtab, NSDB_RPC, "getrpcbynumber_r", defaultsrc,
860 	    number, rpc, buffer, bufsize, &ret_errno);
861 
862 	if (rv == NS_SUCCESS)
863 		return (0);
864 	else
865 		return (ret_errno);
866 }
867 
868 static int
869 getrpcent_r(struct rpcent *rpc, char *buffer, size_t bufsize,
870 	struct rpcent **result)
871 {
872 #ifdef NS_CACHING
873 	static const nss_cache_info cache_info = NS_MP_CACHE_INFO_INITIALIZER(
874 		rpc, (void *)nss_lt_all,
875 		rpc_marshal_func, rpc_unmarshal_func);
876 #endif
877 	static const ns_dtab dtab[] = {
878 		{ NSSRC_FILES, files_rpcent, (void *)nss_lt_all },
879 #ifdef YP
880 		{ NSSRC_NIS, nis_rpcent, (void *)nss_lt_all },
881 #endif
882 #ifdef NS_CACHING
883 		NS_CACHE_CB(&cache_info)
884 #endif
885 		{ NULL, NULL, NULL }
886 	};
887 	int rv, ret_errno;
888 
889 	ret_errno = 0;
890 	*result = NULL;
891 	rv = nsdispatch(result, dtab, NSDB_RPC, "getrpcent_r", defaultsrc,
892 	    rpc, buffer, bufsize, &ret_errno);
893 
894 	if (rv == NS_SUCCESS)
895 		return (0);
896 	else
897 		return (ret_errno);
898 }
899 
900 /* get** wrappers for get**_r functions implementation */
901 static 	void
902 rpcent_endstate(void *p)
903 {
904 	if (p == NULL)
905 		return;
906 
907 	free(((struct rpcent_state *)p)->buffer);
908 	free(p);
909 }
910 
911 static	int
912 wrap_getrpcbyname_r(union key key, struct rpcent *rpc, char *buffer,
913     size_t bufsize, struct rpcent **res)
914 {
915 	return (getrpcbyname_r(key.name, rpc, buffer, bufsize, res));
916 }
917 
918 static	int
919 wrap_getrpcbynumber_r(union key key, struct rpcent *rpc, char *buffer,
920     size_t bufsize, struct rpcent **res)
921 {
922 	return (getrpcbynumber_r(key.number, rpc, buffer, bufsize, res));
923 }
924 
925 static	int
926 wrap_getrpcent_r(union key key __unused, struct rpcent *rpc, char *buffer,
927     size_t bufsize, struct rpcent **res)
928 {
929 	return (getrpcent_r(rpc, buffer, bufsize, res));
930 }
931 
932 static struct rpcent *
933 getrpc(int (*fn)(union key, struct rpcent *, char *, size_t, struct rpcent **),
934     union key key)
935 {
936 	int		 rv;
937 	struct rpcent	*res;
938 	struct rpcent_state * st;
939 
940 	rv=rpcent_getstate(&st);
941 	if (rv != 0) {
942 		errno = rv;
943 		return NULL;
944 	}
945 
946 	if (st->buffer == NULL) {
947 		st->buffer = malloc(RPCENT_STORAGE_INITIAL);
948 		if (st->buffer == NULL)
949 			return (NULL);
950 		st->bufsize = RPCENT_STORAGE_INITIAL;
951 	}
952 	do {
953 		rv = fn(key, &st->rpc, st->buffer, st->bufsize, &res);
954 		if (res == NULL && rv == ERANGE) {
955 			free(st->buffer);
956 			if ((st->bufsize << 1) > RPCENT_STORAGE_MAX) {
957 				st->buffer = NULL;
958 				errno = ERANGE;
959 				return (NULL);
960 			}
961 			st->bufsize <<= 1;
962 			st->buffer = malloc(st->bufsize);
963 			if (st->buffer == NULL)
964 				return (NULL);
965 		}
966 	} while (res == NULL && rv == ERANGE);
967 	if (rv != 0)
968 		errno = rv;
969 
970 	return (res);
971 }
972 
973 struct rpcent *
974 getrpcbyname(char *name)
975 {
976 	union key key;
977 
978 	key.name = name;
979 
980 	return (getrpc(wrap_getrpcbyname_r, key));
981 }
982 
983 struct rpcent *
984 getrpcbynumber(int number)
985 {
986 	union key key;
987 
988 	key.number = number;
989 
990 	return (getrpc(wrap_getrpcbynumber_r, key));
991 }
992 
993 struct rpcent *
994 getrpcent()
995 {
996 	union key key;
997 
998 	key.number = 0;	/* not used */
999 
1000 	return (getrpc(wrap_getrpcent_r, key));
1001 }
1002 
1003 void
1004 setrpcent(int stayopen)
1005 {
1006 #ifdef NS_CACHING
1007 	static const nss_cache_info cache_info = NS_MP_CACHE_INFO_INITIALIZER(
1008 		rpc, (void *)nss_lt_all,
1009 		NULL, NULL);
1010 #endif
1011 
1012 	static const ns_dtab dtab[] = {
1013 		{ NSSRC_FILES, files_setrpcent, (void *)SETRPCENT },
1014 #ifdef YP
1015 		{ NSSRC_NIS, nis_setrpcent, (void *)SETRPCENT },
1016 #endif
1017 #ifdef NS_CACHING
1018 		NS_CACHE_CB(&cache_info)
1019 #endif
1020 		{ NULL, NULL, NULL }
1021 	};
1022 
1023 	(void)nsdispatch(NULL, dtab, NSDB_RPC, "setrpcent", defaultsrc,
1024 		stayopen);
1025 }
1026 
1027 void
1028 endrpcent()
1029 {
1030 #ifdef NS_CACHING
1031 	static const nss_cache_info cache_info = NS_MP_CACHE_INFO_INITIALIZER(
1032 		rpc, (void *)nss_lt_all,
1033 		NULL, NULL);
1034 #endif
1035 
1036 	static const ns_dtab dtab[] = {
1037 		{ NSSRC_FILES, files_setrpcent, (void *)ENDRPCENT },
1038 #ifdef YP
1039 		{ NSSRC_NIS, nis_setrpcent, (void *)ENDRPCENT },
1040 #endif
1041 #ifdef NS_CACHING
1042 		NS_CACHE_CB(&cache_info)
1043 #endif
1044 		{ NULL, NULL, NULL }
1045 	};
1046 
1047 	(void)nsdispatch(NULL, dtab, NSDB_RPC, "endrpcent", defaultsrc);
1048 }
1049