xref: /illumos-gate/usr/src/lib/nsswitch/files/common/files_common.c (revision 66582b606a8194f7f3ba5b3a3a6dca5b0d346361)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 /*
26  * Copyright 2014 Nexenta Systems, Inc.  All rights reserved.
27  * Copyright (c) 2017, Joyent, Inc.
28  */
29 
30 /*
31  * Common code and structures used by name-service-switch "files" backends.
32  */
33 
34 /*
35  * An implementation that used mmap() sensibly would be a wonderful thing,
36  *   but this here is just yer standard fgets() thang.
37  */
38 
39 #include "files_common.h"
40 #include <stdio.h>
41 #include <stdlib.h>
42 #include <string.h>
43 #include <ctype.h>
44 #include <fcntl.h>
45 #include <poll.h>
46 #include <unistd.h>
47 #include <sys/stat.h>
48 #include <sys/mman.h>
49 
50 /*ARGSUSED*/
51 nss_status_t
52 _nss_files_setent(be, dummy)
53 	files_backend_ptr_t	be;
54 	void			*dummy;
55 {
56 	if (be->f == 0) {
57 		if (be->filename == 0) {
58 			/* Backend isn't initialized properly? */
59 			return (NSS_UNAVAIL);
60 		}
61 		if ((be->f = fopen(be->filename, "rF")) == 0) {
62 			return (NSS_UNAVAIL);
63 		}
64 	} else {
65 		rewind(be->f);
66 	}
67 	return (NSS_SUCCESS);
68 }
69 
70 /*ARGSUSED*/
71 nss_status_t
72 _nss_files_endent(be, dummy)
73 	files_backend_ptr_t	be;
74 	void			*dummy;
75 {
76 	if (be->f != 0) {
77 		(void) fclose(be->f);
78 		be->f = 0;
79 	}
80 	if (be->buf != 0) {
81 		free(be->buf);
82 		be->buf = 0;
83 	}
84 	return (NSS_SUCCESS);
85 }
86 
87 /*
88  * This routine reads a line, including the processing of continuation
89  * characters.  It always leaves (or inserts) \n\0 at the end of the line.
90  * It returns the length of the line read, excluding the \n\0.  Who's idea
91  * was this?
92  * Returns -1 on EOF.
93  *
94  * Note that since each concurrent call to _nss_files_read_line has
95  * it's own FILE pointer, we can use getc_unlocked w/o difficulties,
96  * a substantial performance win.
97  */
98 int
99 _nss_files_read_line(f, buffer, buflen)
100 	FILE			*f;
101 	char			*buffer;
102 	int			buflen;
103 {
104 	int			linelen;	/* 1st unused slot in buffer */
105 	int			c;
106 
107 	/*CONSTCOND*/
108 	while (1) {
109 		linelen = 0;
110 		while (linelen < buflen - 1) {	/* "- 1" saves room for \n\0 */
111 			switch (c = getc_unlocked(f)) {
112 			case EOF:
113 				if (linelen == 0 ||
114 				    buffer[linelen - 1] == '\\') {
115 					return (-1);
116 				} else {
117 					buffer[linelen    ] = '\n';
118 					buffer[linelen + 1] = '\0';
119 					return (linelen);
120 				}
121 			case '\n':
122 				if (linelen > 0 &&
123 				    buffer[linelen - 1] == '\\') {
124 					--linelen;  /* remove the '\\' */
125 				} else {
126 					buffer[linelen    ] = '\n';
127 					buffer[linelen + 1] = '\0';
128 					return (linelen);
129 				}
130 				break;
131 			default:
132 				buffer[linelen++] = c;
133 			}
134 		}
135 		/* Buffer overflow -- eat rest of line and loop again */
136 		/* ===> Should syslog() */
137 		do {
138 			c = getc_unlocked(f);
139 			if (c == EOF) {
140 				return (-1);
141 			}
142 		} while (c != '\n');
143 	}
144 	/*NOTREACHED*/
145 }
146 
147 /*
148  * used only for getgroupbymem() now.
149  */
150 nss_status_t
151 _nss_files_do_all(be, args, filter, func)
152 	files_backend_ptr_t	be;
153 	void			*args;
154 	const char		*filter;
155 	files_do_all_func_t	func;
156 {
157 	long			grlen;
158 	char			*buffer;
159 	int			buflen;
160 	nss_status_t		res;
161 
162 	if (be->buf == 0) {
163 		if ((grlen = sysconf(_SC_GETGR_R_SIZE_MAX)) > 0)
164 			be->minbuf = grlen;
165 		if ((be->buf = malloc(be->minbuf)) == 0)
166 			return (NSS_UNAVAIL);
167 	}
168 	buffer = be->buf;
169 	buflen = be->minbuf;
170 
171 	if ((res = _nss_files_setent(be, 0)) != NSS_SUCCESS) {
172 		return (res);
173 	}
174 
175 	res = NSS_NOTFOUND;
176 
177 	do {
178 		int		linelen;
179 
180 		if ((linelen = _nss_files_read_line(be->f, buffer,
181 		    buflen)) < 0) {
182 			/* End of file */
183 			break;
184 		}
185 		if (filter != 0 && strstr(buffer, filter) == 0) {
186 			/*
187 			 * Optimization:  if the entry doesn't contain the
188 			 *   filter string then it can't be the entry we want,
189 			 *   so don't bother looking more closely at it.
190 			 */
191 			continue;
192 		}
193 		res = (*func)(buffer, linelen, args);
194 
195 	} while (res == NSS_NOTFOUND);
196 
197 	(void) _nss_files_endent(be, 0);
198 	return (res);
199 }
200 
201 /*
202  * Could implement this as an iterator function on top of _nss_files_do_all(),
203  *   but the shared code is small enough that it'd be pretty silly.
204  */
205 nss_status_t
206 _nss_files_XY_all(be, args, netdb, filter, check)
207 	files_backend_ptr_t	be;
208 	nss_XbyY_args_t		*args;
209 	int			netdb;		/* whether it uses netdb */
210 						/* format or not */
211 	const char		*filter;	/* advisory, to speed up */
212 						/* string search */
213 	files_XY_check_func	check;	/* NULL means one-shot, for getXXent */
214 {
215 	char			*r;
216 	nss_status_t		res;
217 	int	parsestat;
218 	int (*func)();
219 
220 	if (filter != NULL && *filter == '\0')
221 		return (NSS_NOTFOUND);
222 	if (be->buf == 0 || (be->minbuf < args->buf.buflen)) {
223 		if (be->minbuf < args->buf.buflen) {
224 			if (be->buf == 0) {
225 				be->minbuf = args->buf.buflen;
226 			} else if (
227 			    (r = realloc(be->buf, args->buf.buflen)) != NULL) {
228 				be->buf = r;
229 				be->minbuf = args->buf.buflen;
230 			}
231 		}
232 		if (be->buf == 0 &&
233 			(be->buf = malloc(be->minbuf)) == 0)
234 				return (NSS_UNAVAIL);
235 	}
236 
237 	if (check != 0 || be->f == 0) {
238 		if ((res = _nss_files_setent(be, 0)) != NSS_SUCCESS) {
239 			return (res);
240 		}
241 	}
242 
243 	res = NSS_NOTFOUND;
244 
245 	/*CONSTCOND*/
246 	while (1) {
247 		char		*instr	= be->buf;
248 		int		linelen;
249 
250 		if ((linelen = _nss_files_read_line(be->f, instr,
251 		    be->minbuf)) < 0) {
252 			/* End of file */
253 			args->returnval = 0;
254 			args->returnlen = 0;
255 			break;
256 		}
257 		if (filter != 0 && strstr(instr, filter) == 0) {
258 			/*
259 			 * Optimization:  if the entry doesn't contain the
260 			 *   filter string then it can't be the entry we want,
261 			 *   so don't bother looking more closely at it.
262 			 */
263 			continue;
264 		}
265 		if (netdb) {
266 			char		*first;
267 			char		*last;
268 
269 			if ((last = strchr(instr, '#')) == 0) {
270 				last = instr + linelen;
271 			}
272 			*last-- = '\0';		/* Nuke '\n' or #comment */
273 
274 			/*
275 			 * Skip leading whitespace.  Normally there isn't
276 			 *   any, so it's not worth calling strspn().
277 			 */
278 			for (first = instr;  isspace(*first);  first++) {
279 				;
280 			}
281 			if (*first == '\0') {
282 				continue;
283 			}
284 			/*
285 			 * Found something non-blank on the line.  Skip back
286 			 * over any trailing whitespace;  since we know
287 			 * there's non-whitespace earlier in the line,
288 			 * checking for termination is easy.
289 			 */
290 			while (isspace(*last)) {
291 				--last;
292 			}
293 
294 			linelen = last - first + 1;
295 			if (first != instr) {
296 					instr = first;
297 			}
298 		}
299 
300 		args->returnval = 0;
301 		args->returnlen = 0;
302 
303 		if (check != NULL && (*check)(args, instr, linelen) == 0)
304 			continue;
305 
306 		parsestat = NSS_STR_PARSE_SUCCESS;
307 		if (be->filename != NULL) {
308 			/*
309 			 * Special case for passwd and group wherein we
310 			 * replace uids/gids > MAXUID by ID_NOBODY
311 			 * because files backend does not support
312 			 * ephemeral ids.
313 			 */
314 			if (strcmp(be->filename, PF_PATH) == 0)
315 				parsestat = validate_passwd_ids(instr,
316 				    &linelen, be->minbuf, 2);
317 			else if (strcmp(be->filename, GF_PATH) == 0)
318 				parsestat = validate_group_ids(instr,
319 				    &linelen, be->minbuf, 2, check);
320 		}
321 
322 		if (parsestat == NSS_STR_PARSE_SUCCESS) {
323 			func = args->str2ent;
324 			parsestat = (*func)(instr, linelen, args->buf.result,
325 			    args->buf.buffer, args->buf.buflen);
326 		}
327 
328 		if (parsestat == NSS_STR_PARSE_SUCCESS) {
329 			args->returnval = (args->buf.result != NULL)?
330 					args->buf.result : args->buf.buffer;
331 			args->returnlen = linelen;
332 			res = NSS_SUCCESS;
333 			break;
334 		} else if (parsestat == NSS_STR_PARSE_ERANGE) {
335 			args->erange = 1;
336 			break;
337 		} /* else if (parsestat == NSS_STR_PARSE_PARSE) don't care ! */
338 	}
339 
340 	/*
341 	 * stayopen is set to 0 by default in order to close the opened
342 	 * file.  Some applications may break if it is set to 1.
343 	 */
344 	if (check != 0 && !args->stayopen) {
345 		(void) _nss_files_endent(be, 0);
346 	}
347 
348 	return (res);
349 }
350 
351 /*
352  * File hashing support.  Critical for sites with large (e.g. 1000+ lines)
353  * /etc/passwd or /etc/group files.  Currently only used by getpw*() and
354  * getgr*() routines, but any files backend can use this stuff.
355  */
356 static void
357 _nss_files_hash_destroy(files_hash_t *fhp)
358 {
359 	free(fhp->fh_table);
360 	fhp->fh_table = NULL;
361 	free(fhp->fh_line);
362 	fhp->fh_line = NULL;
363 	free(fhp->fh_file_start);
364 	fhp->fh_file_start = NULL;
365 }
366 #ifdef PIC
367 /*
368  * It turns out the hashing stuff really needs to be disabled for processes
369  * other than the nscd; the consumption of swap space and memory is otherwise
370  * unacceptable when the nscd is killed w/ a large passwd file (4M) active.
371  * See 4031930 for details.
372  * So we just use this psuedo function to enable the hashing feature.  Since
373  * this function name is private, we just create a function w/ the name
374  *  __nss_use_files_hash in the nscd itself and everyone else uses the old
375  * interface.
376  * We also disable hashing for .a executables to avoid problems with large
377  * files....
378  */
379 
380 #pragma weak __nss_use_files_hash
381 
382 extern void  __nss_use_files_hash(void);
383 #endif /* pic */
384 
385 /*ARGSUSED*/
386 nss_status_t
387 _nss_files_XY_hash(files_backend_ptr_t be, nss_XbyY_args_t *args,
388 	int netdb, files_hash_t *fhp, int hashop, files_XY_check_func check)
389 {
390 	/* LINTED E_FUNC_VAR_UNUSED */
391 	int fd, retries, ht, stat;
392 	/* LINTED E_FUNC_VAR_UNUSED */
393 	uint_t hash, line, f;
394 	/* LINTED E_FUNC_VAR_UNUSED */
395 	files_hashent_t *hp, *htab;
396 	/* LINTED E_FUNC_VAR_UNUSED */
397 	char *cp, *first, *last;
398 	/* LINTED E_FUNC_VAR_UNUSED */
399 	nss_XbyY_args_t xargs;
400 	/* LINTED E_FUNC_VAR_UNUSED */
401 	struct stat64 st;
402 
403 #ifndef PIC
404 	return (_nss_files_XY_all(be, args, netdb, 0, check));
405 }
406 #else
407 	if (__nss_use_files_hash == 0)
408 		return (_nss_files_XY_all(be, args, netdb, 0, check));
409 
410 	mutex_lock(&fhp->fh_lock);
411 retry:
412 	retries = 100;
413 	while (stat64(be->filename, &st) < 0) {
414 		/*
415 		 * This can happen only in two cases: Either the file is
416 		 * completely missing and we were not able to read it yet
417 		 * (fh_table is NULL), or there is some brief period when the
418 		 * file is being modified/renamed.  Keep trying until things
419 		 * settle down, but eventually give up.
420 		 */
421 		if (fhp->fh_table == NULL || --retries == 0)
422 			goto unavail;
423 		poll(0, 0, 100);
424 	}
425 
426 	if (st.st_mtim.tv_sec == fhp->fh_mtime.tv_sec &&
427 	    st.st_mtim.tv_nsec == fhp->fh_mtime.tv_nsec &&
428 	    fhp->fh_table != NULL) {
429 		htab = &fhp->fh_table[hashop * fhp->fh_size];
430 		hash = fhp->fh_hash_func[hashop](args, 1, NULL, 0);
431 		for (hp = htab[hash % fhp->fh_size].h_first; hp != NULL;
432 		    hp = hp->h_next) {
433 			if (hp->h_hash != hash)
434 				continue;
435 			line = hp - htab;
436 			if ((*check)(args, fhp->fh_line[line].l_start,
437 					fhp->fh_line[line].l_len) == 0)
438 				continue;
439 
440 			if (be->filename != NULL) {
441 				stat = NSS_STR_PARSE_SUCCESS;
442 				if (strcmp(be->filename, PF_PATH) == 0)
443 					stat = validate_passwd_ids(
444 					    fhp->fh_line[line].l_start,
445 					    &fhp->fh_line[line].l_len,
446 					    fhp->fh_line[line].l_len + 1,
447 					    1);
448 				else if (strcmp(be->filename, GF_PATH) == 0)
449 					stat = validate_group_ids(
450 					    fhp->fh_line[line].l_start,
451 					    &fhp->fh_line[line].l_len,
452 					    fhp->fh_line[line].l_len + 1,
453 					    1, check);
454 				if (stat != NSS_STR_PARSE_SUCCESS) {
455 					if (stat == NSS_STR_PARSE_ERANGE)
456 						args->erange = 1;
457 					continue;
458 				}
459 			}
460 
461 			if ((*args->str2ent)(fhp->fh_line[line].l_start,
462 			    fhp->fh_line[line].l_len, args->buf.result,
463 			    args->buf.buffer, args->buf.buflen) ==
464 			    NSS_STR_PARSE_SUCCESS) {
465 				args->returnval = (args->buf.result)?
466 					args->buf.result:args->buf.buffer;
467 				args->returnlen = fhp->fh_line[line].l_len;
468 				mutex_unlock(&fhp->fh_lock);
469 				return (NSS_SUCCESS);
470 			} else {
471 				args->erange = 1;
472 			}
473 		}
474 		args->returnval = 0;
475 		args->returnlen = 0;
476 		mutex_unlock(&fhp->fh_lock);
477 		return (NSS_NOTFOUND);
478 	}
479 
480 	_nss_files_hash_destroy(fhp);
481 
482 	if (st.st_size > SSIZE_MAX)
483 		goto unavail;
484 
485 	if ((fhp->fh_file_start = malloc((ssize_t)st.st_size + 1)) == NULL)
486 		goto unavail;
487 
488 	if ((fd = open(be->filename, O_RDONLY)) < 0)
489 		goto unavail;
490 
491 	if (read(fd, fhp->fh_file_start, (ssize_t)st.st_size) !=
492 	    (ssize_t)st.st_size) {
493 		close(fd);
494 		goto retry;
495 	}
496 
497 	close(fd);
498 
499 	fhp->fh_file_end = fhp->fh_file_start + (off_t)st.st_size;
500 	*fhp->fh_file_end = '\n';
501 	fhp->fh_mtime = st.st_mtim;
502 
503 	/*
504 	 * If the file changed since we read it, or if it's less than
505 	 * 1-2 seconds old, don't trust it; its modification may still
506 	 * be in progress.  The latter is a heuristic hack to minimize
507 	 * the likelihood of damage if someone modifies /etc/mumble
508 	 * directly (as opposed to editing and renaming a temp file).
509 	 *
510 	 * Note: the cast to u_int is there in case (1) someone rdated
511 	 * the system backwards since the last modification of /etc/mumble
512 	 * or (2) this is a diskless client whose time is badly out of sync
513 	 * with its server.  The 1-2 second age hack doesn't cover these
514 	 * cases -- oh well.
515 	 */
516 	if (stat64(be->filename, &st) < 0 ||
517 	    st.st_mtim.tv_sec != fhp->fh_mtime.tv_sec ||
518 	    st.st_mtim.tv_nsec != fhp->fh_mtime.tv_nsec ||
519 	    (uint_t)(time(0) - st.st_mtim.tv_sec + 2) < 4) {
520 		poll(0, 0, 1000);
521 		goto retry;
522 	}
523 
524 	line = 1;
525 	for (cp = fhp->fh_file_start; cp < fhp->fh_file_end; cp++)
526 		if (*cp == '\n')
527 			line++;
528 
529 	for (f = 2; f * f <= line; f++) {	/* find next largest prime */
530 		if (line % f == 0) {
531 			f = 1;
532 			line++;
533 		}
534 	}
535 
536 	fhp->fh_size = line;
537 	fhp->fh_line = malloc(line * sizeof (files_linetab_t));
538 	fhp->fh_table = calloc(line * fhp->fh_nhtab, sizeof (files_hashent_t));
539 	if (fhp->fh_line == NULL || fhp->fh_table == NULL)
540 		goto unavail;
541 
542 	line = 0;
543 	cp = fhp->fh_file_start;
544 	while (cp < fhp->fh_file_end) {
545 		first = cp;
546 		while (*cp != '\n')
547 			cp++;
548 		if (cp > first && *(cp - 1) == '\\') {
549 			memmove(first + 2, first, cp - first - 1);
550 			cp = first + 2;
551 			continue;
552 		}
553 		last = cp;
554 		*cp++ = '\0';
555 		if (netdb) {
556 			if ((last = strchr(first, '#')) == 0)
557 				last = cp - 1;
558 			*last-- = '\0';		/* nuke '\n' or #comment */
559 			while (isspace(*first))	/* nuke leading whitespace */
560 				first++;
561 			if (*first == '\0')	/* skip content-free lines */
562 				continue;
563 			while (isspace(*last))	/* nuke trailing whitespace */
564 				--last;
565 			*++last = '\0';
566 		}
567 		for (ht = 0; ht < fhp->fh_nhtab; ht++) {
568 			hp = &fhp->fh_table[ht * fhp->fh_size + line];
569 			hp->h_hash = fhp->fh_hash_func[ht](&xargs, 0, first,
570 					last - first);
571 		}
572 		fhp->fh_line[line].l_start = first;
573 		fhp->fh_line[line++].l_len = last - first;
574 	}
575 
576 	/*
577 	 * Populate the hash tables in reverse order so that the hash chains
578 	 * end up in forward order.  This ensures that hashed lookups find
579 	 * things in the same order that a linear search of the file would.
580 	 * This is essential in cases where there could be multiple matches.
581 	 * For example: until 2.7, root and smtp both had uid 0; but we
582 	 * certainly wouldn't want getpwuid(0) to return smtp.
583 	 */
584 	for (ht = 0; ht < fhp->fh_nhtab; ht++) {
585 		htab = &fhp->fh_table[ht * fhp->fh_size];
586 		for (hp = &htab[line - 1]; hp >= htab; hp--) {
587 			uint_t bucket = hp->h_hash % fhp->fh_size;
588 			hp->h_next = htab[bucket].h_first;
589 			htab[bucket].h_first = hp;
590 		}
591 	}
592 
593 	goto retry;
594 
595 unavail:
596 	_nss_files_hash_destroy(fhp);
597 	mutex_unlock(&fhp->fh_lock);
598 	return (NSS_UNAVAIL);
599 }
600 #endif /* PIC */
601 
602 nss_status_t
603 _nss_files_getent_rigid(be, a)
604 	files_backend_ptr_t	be;
605 	void			*a;
606 {
607 	nss_XbyY_args_t		*args = (nss_XbyY_args_t *)a;
608 
609 	return (_nss_files_XY_all(be, args, 0, 0, 0));
610 }
611 
612 nss_status_t
613 _nss_files_getent_netdb(be, a)
614 	files_backend_ptr_t	be;
615 	void			*a;
616 {
617 	nss_XbyY_args_t		*args = (nss_XbyY_args_t *)a;
618 
619 	return (_nss_files_XY_all(be, args, 1, 0, 0));
620 }
621 
622 /*ARGSUSED*/
623 nss_status_t
624 _nss_files_destr(be, dummy)
625 	files_backend_ptr_t	be;
626 	void			*dummy;
627 {
628 	if (be != 0) {
629 		if (be->f != 0) {
630 			(void) _nss_files_endent(be, 0);
631 		}
632 		if (be->hashinfo != NULL) {
633 			(void) mutex_lock(&be->hashinfo->fh_lock);
634 			if (--be->hashinfo->fh_refcnt == 0)
635 				_nss_files_hash_destroy(be->hashinfo);
636 			(void) mutex_unlock(&be->hashinfo->fh_lock);
637 		}
638 		free(be);
639 	}
640 	return (NSS_SUCCESS);	/* In case anyone is dumb enough to check */
641 }
642 
643 nss_backend_t *
644 _nss_files_constr(ops, n_ops, filename, min_bufsize, fhp)
645 	files_backend_op_t	ops[];
646 	int			n_ops;
647 	const char		*filename;
648 	int			min_bufsize;
649 	files_hash_t		*fhp;
650 {
651 	files_backend_ptr_t	be;
652 
653 	if ((be = (files_backend_ptr_t)malloc(sizeof (*be))) == 0) {
654 		return (0);
655 	}
656 	be->ops		= ops;
657 	be->n_ops	= n_ops;
658 	be->filename	= filename;
659 	be->minbuf	= min_bufsize;
660 	be->f		= 0;
661 	be->buf		= 0;
662 	be->hashinfo	= fhp;
663 
664 	if (fhp != NULL) {
665 		(void) mutex_lock(&fhp->fh_lock);
666 		fhp->fh_refcnt++;
667 		(void) mutex_unlock(&fhp->fh_lock);
668 	}
669 
670 	return ((nss_backend_t *)be);
671 }
672 
673 int
674 _nss_files_check_name_colon(nss_XbyY_args_t *argp, const char *line,
675 	int linelen)
676 {
677 	const char	*linep, *limit;
678 	const char	*keyp = argp->key.name;
679 
680 	linep = line;
681 	limit = line + linelen;
682 	while (*keyp && linep < limit && *keyp == *linep) {
683 		keyp++;
684 		linep++;
685 	}
686 	return (linep < limit && *keyp == '\0' && *linep == ':');
687 }
688 
689 /*
690  * This routine is used to parse lines of the form:
691  * 	name number aliases
692  * It returns 1 if the key in argp matches any one of the
693  * names in the line, otherwise 0
694  * Used by rpc, networks, protocols
695  */
696 int
697 _nss_files_check_name_aliases(nss_XbyY_args_t *argp, const char *line,
698 	int linelen)
699 {
700 	const char	*limit, *linep, *keyp;
701 
702 	linep = line;
703 	limit = line + linelen;
704 	keyp = argp->key.name;
705 
706 	/* compare name */
707 	while (*keyp && linep < limit && !isspace(*linep) && *keyp == *linep) {
708 		keyp++;
709 		linep++;
710 	}
711 	if (*keyp == '\0' && linep < limit && isspace(*linep))
712 		return (1);
713 	/* skip remainder of the name, if any */
714 	while (linep < limit && !isspace(*linep))
715 		linep++;
716 	/* skip the delimiting spaces */
717 	while (linep < limit && isspace(*linep))
718 		linep++;
719 	/* compare with the aliases */
720 	while (linep < limit) {
721 		/*
722 		 * 1st pass: skip number
723 		 * Other passes: skip remainder of the alias name, if any
724 		 */
725 		while (linep < limit && !isspace(*linep))
726 			linep++;
727 		/* skip the delimiting spaces */
728 		while (linep < limit && isspace(*linep))
729 			linep++;
730 		/* compare with the alias name */
731 		keyp = argp->key.name;
732 		while (*keyp && linep < limit && !isspace(*linep) &&
733 		    *keyp == *linep) {
734 			keyp++;
735 			linep++;
736 		}
737 		if (*keyp == '\0' && (linep == limit || isspace(*linep)))
738 			return (1);
739 	}
740 	return (0);
741 }
742 
743 /*
744  * A few NSS modules hold onto data for the duration of their module. In this
745  * case, when that module goes away, we must free that data. This is a place
746  * that allows for modules to register items to take care of.
747  */
748 #pragma fini(_nss_files_fini)
749 static void
750 _nss_files_fini(void)
751 {
752 	getexecattr_fini();
753 }
754