xref: /illumos-gate/usr/src/lib/libnisdb/ldap_val.c (revision 796b8631498f69a3e21b5c35aee280499f64420e)
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 2015 Gary Mills
23  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
24  * Use is subject to license terms.
25  */
26 
27 
28 #include <lber.h>
29 #include <ldap.h>
30 #include <strings.h>
31 #include <errno.h>
32 
33 #include "nisdb_mt.h"
34 
35 #include "ldap_util.h"
36 #include "ldap_op.h"
37 #include "ldap_ruleval.h"
38 #include "ldap_attr.h"
39 #include "ldap_val.h"
40 #include "ldap_ldap.h"
41 
42 extern int yp2ldap;
43 
44 
45 __nis_mapping_format_t *
46 cloneMappingFormat(__nis_mapping_format_t *m) {
47 	__nis_mapping_format_t	*new;
48 	int			i, nf, err;
49 	char			*myself = "cloneMappingFormat";
50 
51 	if (m == 0)
52 		return (0);
53 
54 	for (nf = 0; m[nf].type != mmt_end; nf++);
55 	nf++;
56 
57 	new = am(myself, nf * sizeof (new[0]));
58 	if (new == 0)
59 		return (0);
60 
61 	/* Copy the whole array */
62 	memcpy(new, m, nf * sizeof (new[0]));
63 
64 	/* Make copies of allocated stuff */
65 	for (i = 0, err = 0; i < nf; i++) {
66 		switch (m[i].type) {
67 		case mmt_string:
68 			new[i].match.string = sdup(myself, T,
69 							m[i].match.string);
70 			if (new[i].match.string == 0 && m[i].match.string != 0)
71 				err++;
72 			break;
73 		case mmt_single:
74 			new[i].match.single.lo =
75 				am(myself, m[i].match.single.numRange *
76 					sizeof (new[i].match.single.lo[0]));
77 			new[i].match.single.hi =
78 				am(myself, m[i].match.single.numRange *
79 					sizeof (new[i].match.single.hi[0]));
80 			if (new[i].match.single.lo != 0)
81 				memcpy(new[i].match.single.lo,
82 					m[i].match.single.lo,
83 					m[i].match.single.numRange);
84 			else if (m[i].match.single.lo != 0)
85 				err++;
86 			if (new[i].match.single.hi != 0)
87 				memcpy(new[i].match.single.hi,
88 					m[i].match.single.hi,
89 					m[i].match.single.numRange);
90 			else if (m[i].match.single.hi != 0)
91 				err++;
92 			break;
93 		case mmt_berstring:
94 			new[i].match.berString = sdup(myself, T,
95 							m[i].match.berString);
96 			if (new[i].match.berString == 0 &&
97 					m[i].match.berString != 0)
98 				err++;
99 			break;
100 		case mmt_item:
101 		case mmt_limit:
102 		case mmt_any:
103 		case mmt_begin:
104 		case mmt_end:
105 		default:
106 			break;
107 		}
108 	}
109 
110 	/* If there were memory allocation errors, free the copy */
111 	if (err > 0) {
112 		freeMappingFormat(new);
113 		new = 0;
114 	}
115 
116 	return (new);
117 }
118 
119 void
120 freeMappingFormat(__nis_mapping_format_t *m) {
121 	int	i;
122 
123 	if (m == 0)
124 		return;
125 
126 	for (i = 0; m[i].type != mmt_end; i++) {
127 		switch (m[i].type) {
128 		case mmt_string:
129 			sfree(m[i].match.string);
130 			break;
131 		case mmt_single:
132 			sfree(m[i].match.single.lo);
133 			sfree(m[i].match.single.hi);
134 			break;
135 		case mmt_berstring:
136 			sfree(m[i].match.berString);
137 			break;
138 		case mmt_item:
139 		case mmt_limit:
140 		case mmt_any:
141 		case mmt_begin:
142 		case mmt_end:
143 		default:
144 			break;
145 		}
146 	}
147 
148 	free(m);
149 }
150 
151 
152 void
153 copyIndex(__nis_index_t *old, __nis_index_t *new, int *err) {
154 	int	i;
155 	char	*myself = "copyIndex";
156 
157 	if (old == 0 || new == 0) {
158 		*err = EINVAL;
159 		return;
160 	}
161 
162 	for (i = 0; i < old->numIndexes; i++) {
163 		new->name[i] = sdup(myself, T, old->name[i]);
164 		if (new->name[i] == 0 && old->name[i] != 0) {
165 			*err = ENOMEM;
166 			return;
167 		}
168 		new->value[i] = cloneMappingFormat(old->value[i]);
169 		if (new->value[i] == 0 && old->value[i] != 0) {
170 			*err = ENOMEM;
171 			return;
172 		}
173 	}
174 
175 	new->numIndexes = old->numIndexes;
176 }
177 
178 __nis_index_t *
179 cloneIndex(__nis_index_t *old) {
180 	char		*myself = "cloneIndex";
181 	int		err = 0;
182 	__nis_index_t	*new = am(myself, sizeof (*new));
183 
184 	if (old == 0)
185 		return (0);
186 
187 	if (new != 0) {
188 		copyIndex(old, new, &err);
189 		if (err != 0) {
190 			freeIndex(new, 1);
191 			new = 0;
192 		}
193 	}
194 
195 	return (new);
196 }
197 
198 void
199 freeIndex(__nis_index_t *old, bool_t doFree) {
200 	int	i;
201 
202 	if (old == 0)
203 		return;
204 
205 	for (i = 0; i < old->numIndexes; i++) {
206 		sfree(old->name[i]);
207 		freeMappingFormat(old->value[i]);
208 	}
209 
210 	if (doFree)
211 		free(old);
212 }
213 
214 char **
215 cloneName(char **name, int numNames) {
216 	char	**new;
217 	int	i;
218 	char	*myself = "cloneName";
219 
220 	if (name == 0 || numNames <= 0)
221 		return (0);
222 
223 	new = am(myself, numNames * sizeof (new[0]));
224 	if (new == 0)
225 		return (0);
226 
227 	for (i = 0; i < numNames; i++) {
228 		if (name[i] != 0) {
229 			new[i] = sdup(myself, T, name[i]);
230 			if (new[i] == 0) {
231 				for (i--; i >= 0; i--) {
232 					sfree(new[i]);
233 				}
234 				sfree(new);
235 				return (0);
236 			}
237 		} else {
238 			new[i] = 0;
239 		}
240 	}
241 
242 	return (new);
243 }
244 
245 void
246 freeValue(__nis_value_t *val, int count) {
247 	int	c, i;
248 
249 	if (val == 0)
250 		return;
251 
252 	for (c = 0; c < count; c++) {
253 		if (val[c].val != 0) {
254 			for (i = 0; i < val[c].numVals; i++) {
255 				sfree(val[c].val[i].value);
256 			}
257 			free(val[c].val);
258 		}
259 	}
260 
261 	free(val);
262 }
263 
264 __nis_value_t *
265 cloneValue(__nis_value_t *val, int count) {
266 	__nis_value_t	*n;
267 	int		c, i;
268 	char		*myself = "cloneValue";
269 
270 	if (count <= 0 || val == 0)
271 		return (0);
272 
273 	n = am(myself, count * sizeof (*n));
274 	if (n == 0)
275 		return (0);
276 
277 	for (c = 0; c < count; c++) {
278 		n[c].type = val[c].type;
279 		n[c].repeat = val[c].repeat;
280 		n[c].numVals = val[c].numVals;
281 		if (n[c].numVals > 0) {
282 			n[c].val = am(myself, n[c].numVals *
283 						sizeof (n[c].val[0]));
284 			if (n[c].val == 0) {
285 				freeValue(n, c);
286 				return (0);
287 			}
288 		} else {
289 			n[c].val = 0;
290 		}
291 		for (i = 0; i < n[c].numVals; i++) {
292 			int	amlen = val[c].val[i].length;
293 
294 			/*
295 			 * The functions that create string values try to
296 			 * make sure that there's a NUL at the end. However,
297 			 * both NIS+ and LDAP have a tendency to store strings
298 			 * without a NUL, so the value length may not include
299 			 * the NUL (even though it's there). In order to
300 			 * preserve that NUL, we add a byte to the length if
301 			 * the type is vt_string, and there isn't already a
302 			 * NUL at the end. The memory allocation function
303 			 * (am()) will take care of actually putting the NUL
304 			 * in place, since it allocates zero-initialized
305 			 * memory.
306 			 */
307 			n[c].val[i].length = val[c].val[i].length;
308 			if (n[c].type == vt_string && amlen > 0 &&
309 				((char *)val[c].val[i].value)[amlen-1] !=
310 					'\0') {
311 				amlen++;
312 			}
313 			n[c].val[i].value = am(myself, amlen);
314 			if (amlen > 0 && n[c].val[i].value == 0) {
315 				freeValue(n, c);
316 				return (0);
317 			}
318 			memcpy(n[c].val[i].value, val[c].val[i].value,
319 				n[c].val[i].length);
320 		}
321 	}
322 
323 	return (n);
324 }
325 
326 /* Define LBER_USE_DER per ber_decode(3LDAP) */
327 #ifndef	LBER_USE_DER
328 #define	LBER_USE_DER	0x01
329 #endif	/* LBER_USE_DER */
330 
331 /*
332  * Return a copy of 'valIn' where each value has been replaced by the
333  * BER encoded equivalent specified by 'berstring'. 'valIn' is unchanged.
334  */
335 __nis_value_t *
336 berEncode(__nis_value_t *valIn, char *berstring) {
337 	char		*myself = "berEncode";
338 	__nis_value_t	*val;
339 	int		i;
340 
341 	if (valIn == 0 || berstring == 0)
342 		return (0);
343 
344 	val = cloneValue(valIn, 1);
345 	if (val == 0)
346 		return (0);
347 
348 	for (i = 0; i < val->numVals; i++) {
349 		BerElement	*ber = ber_alloc();
350 		struct berval	*bv = 0;
351 		int		ret;
352 
353 		if (ber == 0) {
354 			logmsg(MSG_NOMEM, LOG_ERR, "%s: ber_alloc() => NULL",
355 				myself);
356 			freeValue(val, 1);
357 			return (0);
358 		}
359 
360 		if ((strcmp("b", berstring) == 0 ||
361 				strcmp("i", berstring) == 0)) {
362 			if (val->val[i].length >= sizeof (int)) {
363 				ret = ber_printf(ber, berstring,
364 					*((int *)(val->val[i].value)));
365 			} else {
366 				ret = -1;
367 			}
368 		} else if (strcmp("B", berstring) == 0) {
369 			ret = ber_printf(ber, berstring,
370 				val->val[i].value,
371 				val->val[i].length * 8);
372 		} else if (strcmp("n", berstring) == 0) {
373 			ret = ber_printf(ber, berstring);
374 		} else if (strcmp("o", berstring) == 0) {
375 			ret = ber_printf(ber, berstring,
376 				val->val[i].value, val->val[i].length);
377 		} else if (strcmp("s", berstring) == 0) {
378 			char	*str = am(myself, val->val[i].length + 1);
379 
380 			if (str != 0) {
381 				ret = ber_printf(ber, berstring, str);
382 				free(str);
383 			} else {
384 				ret = -1;
385 			}
386 		} else {
387 			ret = -1;
388 		}
389 
390 		if (ret == -1) {
391 			reportError(NPL_BERENCODE, "%s: BER encoding error",
392 					myself);
393 			ber_free(ber, 1);
394 			freeValue(val, 1);
395 			return (0);
396 		}
397 
398 		if (ber_flatten(ber, &bv) != 0 || bv == 0) {
399 			reportError(NPL_BERENCODE, "%s: ber_flatten() error",
400 					myself);
401 			ber_free(ber, 1);
402 			freeValue(val, 1);
403 			return (0);
404 		}
405 
406 		sfree(val->val[i].value);
407 		val->val[i].length = bv->bv_len;
408 		val->val[i].value = bv->bv_val;
409 
410 		ber_free(ber, 1);
411 	}
412 
413 	val->type = vt_ber;
414 
415 	return (val);
416 }
417 
418 __nis_value_t *
419 berDecode(__nis_value_t *valIn, char *berstring) {
420 	__nis_value_t	*val;
421 	int		i;
422 	char		*myself = "berDecode";
423 
424 	if (valIn == 0 || berstring == 0)
425 		return (0);
426 
427 	val = cloneValue(valIn, 1);
428 	if (val == 0)
429 		return (0);
430 
431 	for (i = 0; i < val->numVals; i++) {
432 		void		*v = 0;
433 		int		ret, len = 0;
434 		struct berval	bv;
435 		BerElement	*ber;
436 
437 		if (val->val[i].value == 0 || val->val[i].length <= 0)
438 			continue;
439 
440 		bv.bv_val = val->val[i].value;
441 		bv.bv_len = val->val[i].length;
442 		ber = ber_init(&bv);
443 		if (ber == 0) {
444 			reportError(NPL_BERDECODE, "%s: ber_init() error",
445 				myself);
446 			freeValue(val, 1);
447 			return (0);
448 		}
449 
450 		if ((strcmp("b", berstring) == 0 ||
451 				strcmp("i", berstring) == 0)) {
452 			len = sizeof (int);
453 			v = am(myself, len);
454 			if (v != 0) {
455 				ret = ber_scanf(ber, berstring, v);
456 			} else {
457 				ret = -1;
458 			}
459 		} else if (strcmp("B", berstring) == 0) {
460 			long	llen;
461 
462 			ret = ber_scanf(ber, berstring, &v, &llen);
463 			if (ret != -1) {
464 				len = llen/8;
465 			}
466 		} else if (strcmp("n", berstring) == 0) {
467 			ret = 0;
468 		} else if (strcmp("o", berstring) == 0) {
469 			struct berval	*bv = am(myself, sizeof (*bv));
470 
471 			if (bv != 0) {
472 				ret = ber_scanf(ber, "O", &bv);
473 				if (ret != -1 && bv != 0) {
474 					v = bv->bv_val;
475 					len = bv->bv_len;
476 				} else {
477 					ret = -1;
478 				}
479 				/* Only free 'bv' itself */
480 				free(bv);
481 			} else {
482 				ret = -1;
483 			}
484 		} else if (strcmp("s", berstring) == 0) {
485 			ret = ber_scanf(ber, "a", &v);
486 			if (ret != -1) {
487 				len = slen(v);
488 			}
489 		} else {
490 			ret = -1;
491 		}
492 
493 		if (ret == -1) {
494 			reportError(NPL_BERDECODE, "%s: BER decoding error",
495 					myself);
496 			freeValue(val, 1);
497 			return (0);
498 		}
499 
500 		/* Free the old value, and replace it with the decoded one */
501 		sfree(val->val[i].value);
502 		val->val[i].value = v;
503 		val->val[i].length = len;
504 	}
505 
506 	return (val);
507 }
508 
509 /*
510  * Return the value of the specified item.
511  */
512 __nis_value_t *
513 getMappingItemVal(__nis_mapping_item_t *item, __nis_mapping_item_type_t native,
514 		__nis_rule_value_t *rv, char *berstring, int *np_ldap_stat) {
515 	__nis_value_t				*val = 0, *nameVal, *exVal = 0;
516 	int					numName, caseInsens, cmp;
517 	int					i, j, k;
518 	char					**name;
519 	enum {rvOnly, rvThenLookup, lookupOnly}	check;
520 	unsigned char				fromldap = '\0';
521 
522 	if (item == 0)
523 		return (0);
524 
525 	/*
526 	 * First, we decide if we should look for the value in 'rv',
527 	 * directly from NIS+/LDAP, or both.
528 	 */
529 	switch (item->type) {
530 	case mit_nisplus:
531 		/* Do we have a valid index/object spec ? */
532 		if (item->searchSpec.obj.index.numIndexes <= 0 &&
533 				item->searchSpec.obj.name == 0) {
534 			/*
535 			 * No valid index/object. If we have a rule-value,
536 			 * use it. Otherwise, return error.
537 			 */
538 			if (rv != 0) {
539 				name = rv->colName;
540 				nameVal = rv->colVal;
541 				numName = rv->numColumns;
542 				caseInsens = 0;
543 				check = rvOnly;
544 			} else {
545 				return (0);
546 			}
547 		} else {
548 			/*
549 			 * Valid index, so skip the rule-value and do
550 			 * a direct NIS+ lookup.
551 			 */
552 			check = lookupOnly;
553 		}
554 		break;
555 	case mit_ldap:
556 		if (rv != 0) {
557 			name = rv->attrName;
558 			nameVal = rv->attrVal;
559 			numName = rv->numAttrs;
560 			caseInsens = 1;
561 			fromldap = '1';
562 		}
563 		/* Do we have a valid triple ? */
564 		if (item->searchSpec.triple.scope == LDAP_SCOPE_UNKNOWN) {
565 			/*
566 			 * No valid triple. If we have a rule-value, use it.
567 			 * Otherwise, return error.
568 			 */
569 			if (rv != 0) {
570 				check = rvOnly;
571 			} else {
572 				return (0);
573 			}
574 		} else if (item->searchSpec.triple.base == 0 &&
575 				item->searchSpec.triple.scope ==
576 					LDAP_SCOPE_ONELEVEL &&
577 				item->searchSpec.triple.attrs == 0 &&
578 				item->searchSpec.triple.element == 0) {
579 			/*
580 			 * We have a valid triple, but it points to the
581 			 * current LDAP container. Thus, first look in
582 			 * the rule-value; if that fails, perform a direct
583 			 * LDAP lookup.
584 			 */
585 			if (rv != 0) {
586 				check = rvThenLookup;
587 			} else {
588 				check = lookupOnly;
589 			}
590 		} else {
591 			/*
592 			 * Valid triple, and it's not the current container
593 			 * (at least not in the trivial sense). Hence, do
594 			 * a direct LDAP lookup.
595 			 */
596 			check = lookupOnly;
597 		}
598 		break;
599 	default:
600 		return (0);
601 	}
602 
603 	/* Check the rule-value */
604 	if (check == rvOnly || check == rvThenLookup) {
605 		for (i = 0; i < numName; i++) {
606 			if (caseInsens)
607 				cmp = strcasecmp(item->name, name[i]);
608 			else
609 				cmp = strcmp(item->name, name[i]);
610 			if (cmp == 0) {
611 				if (nameVal[i].numVals <= 0)
612 					break;
613 				if (berstring == 0) {
614 					val = cloneValue(&nameVal[i], 1);
615 				} else if (yp2ldap && berstring[0] == 'a') {
616 					val = cloneValue(&nameVal[i], 1);
617 				} else {
618 					val = berDecode(&nameVal[i],
619 						berstring);
620 				}
621 				if (val != 0) {
622 					val->repeat = item->repeat;
623 					/*
624 					 * If value for nis+ column is
625 					 * passed with value, val is
626 					 * manipulated in cloneValue().
627 					 * To decide whether there are
628 					 * enough nis+ column values
629 					 * for rule to produce a value,
630 					 * we need nis+ column values
631 					 * as well as nis_mapping_element
632 					 * from the rule. If we are here,
633 					 * it indicates that the 'val has
634 					 * an valid value for the column
635 					 * item-> name. So set
636 					 * NP_LDAP_MAP_SUCCESS
637 					 * to np_ldap-stat.
638 					 */
639 
640 					if (np_ldap_stat != NULL)
641 						*np_ldap_stat =
642 							NP_LDAP_MAP_SUCCESS;
643 				}
644 				break;
645 			}
646 		}
647 	}
648 
649 	/* Do a direct lookup ? */
650 	if (val == 0 && (check == rvThenLookup || check == lookupOnly)) {
651 		if (item->type == mit_ldap) {
652 			int	err = 0;
653 			__nis_search_triple_t	triple;
654 			char			*baseDN;
655 
656 			/*
657 			 * If item->searchSpec.triple.base is NULL, or ends
658 			 * in a comma, append the current search base from
659 			 * the TSD (put there by an upper layer).
660 			 *
661 			 * Special case for N2L mode:
662 			 * if item->searchSpec.triple.base ends in a comma,
663 			 * the current domain Context is used.
664 			 */
665 			if (yp2ldap && item->searchSpec.triple.base &&
666 				strlen(item->searchSpec.triple.base) > 0) {
667 				baseDN = __nisdb_get_tsd()->domainContext;
668 			} else {
669 				baseDN = __nisdb_get_tsd()->searchBase;
670 			}
671 			triple.base = appendBase(item->searchSpec.triple.base,
672 				baseDN, &err, 0);
673 			if (err == 0) {
674 				triple.scope = item->searchSpec.triple.scope;
675 				triple.attrs = item->searchSpec.triple.attrs;
676 				triple.element =
677 					item->searchSpec.triple.element;
678 				val = lookupLDAP(&triple, item->name, rv, 0,
679 					np_ldap_stat);
680 				fromldap = '1';
681 			} else {
682 				val = 0;
683 			}
684 			sfree(triple.base);
685 		}
686 	}
687 
688 
689 	/* Special processing for NIS to LDAP mode */
690 	if (yp2ldap && val != 0) {
691 
692 		/*
693 		 * Escape special chars from dn before sending to DIT,
694 		 * provided val is not ldap-based
695 		 */
696 		if (fromldap == '\0' && __nisdb_get_tsd()->escapeFlag == '1') {
697 			if (escapeSpecialChars(val) < 0) {
698 				freeValue(val, 1);
699 				return (0);
700 			}
701 		} else if (__nisdb_get_tsd()->escapeFlag == '2') {
702 			/* Remove escape chars from data received from DIT */
703 			(void) removeEscapeChars(val);
704 		}
705 
706 		/*
707 		 * Remove from 'val', any values obtained using
708 		 * the 'removespec' syntax
709 		 */
710 
711 		/* Obtain exVal */
712 		if (item->exItem)
713 			exVal = getMappingItemVal(item->exItem, native, rv,
714 			    berstring, NULL);
715 
716 		/* delete */
717 		if (exVal != 0) {
718 			for (i = 0; i < val->numVals; ) {
719 				for (j = 0; j < exVal->numVals; j++) {
720 					if (sstrncmp(val->val[i].value,
721 							exVal->val[j].value,
722 							MAX(val->val[i].length,
723 							exVal->val[j].length))
724 							== 0)
725 						break;
726 				}
727 				if (j < exVal->numVals) {
728 					sfree(val->val[i].value);
729 					val->val[i].value = 0;
730 					val->val[i].length = 0;
731 					for (k = i; k < val->numVals - 1; k++) {
732 						val->val[k] = val->val[k + 1];
733 						val->val[k + 1].value = 0;
734 						val->val[k + 1].length = 0;
735 					}
736 					val->numVals--;
737 				} else
738 					i++;
739 			}
740 
741 			freeValue(exVal, 1);
742 
743 			/*
744 			 * If val->numVals <= 0, then we have no val to
745 			 * return. So free up stuff.
746 			 */
747 			if (val->numVals <= 0) {
748 				free(val->val);
749 				val->val = 0;
750 				free(val);
751 				return (0);
752 			}
753 		}
754 	}
755 
756 	return (val);
757 }
758 
759 __nis_value_t *
760 getMappingFormat(__nis_mapping_format_t *f, __nis_rule_value_t *rv,
761 			__nis_format_arg_t at, void *a, int *numArg) {
762 	char		*myself = "getMappingFormat";
763 	__nis_value_t	*val = 0;
764 	__nis_buffer_t	b = {0, 0};
765 	int		i;
766 
767 	if (f == 0)
768 		return (0);
769 
770 	if (rv == 0) {
771 		val = am(myself, sizeof (*val));
772 		if (val == 0)
773 			return (0);
774 
775 		switch (f->type) {
776 		case mmt_item:
777 			bp2buf(myself, &b, "%%s");
778 			break;
779 		case mmt_string:
780 			bp2buf(myself, &b, "%s", NIL(f->match.string));
781 			break;
782 		case mmt_single:
783 			bp2buf(myself, &b, "[");
784 			for (i = 0; i < f->match.single.numRange; i++) {
785 				if (f->match.single.lo[i] ==
786 						f->match.single.hi[i])
787 					bp2buf(myself, &b, "%c",
788 						f->match.single.lo[i]);
789 				else
790 					bp2buf(myself, &b, "%c-%c",
791 						f->match.single.lo[i],
792 						f->match.single.hi[i]);
793 			}
794 			bp2buf(myself, &b, "]");
795 			break;
796 		case mmt_limit:
797 			break;
798 		case mmt_any:
799 			bp2buf(myself, &b, "*");
800 			break;
801 		case mmt_berstring:
802 			bp2buf(myself, &b, "%s", NIL(f->match.berString));
803 			break;
804 		case mmt_begin:
805 		case mmt_end:
806 			bp2buf(myself, &b, "\"");
807 			break;
808 		default:
809 			bp2buf(myself, &b, "<unknown>");
810 		}
811 		val->type = vt_string;
812 		val->numVals = 1;
813 		val->val = am(myself, sizeof (val->val[0]));
814 		if (val->val == 0) {
815 			sfree(val);
816 			return (0);
817 		}
818 		val->val[0].value = b.buf;
819 		val->val[0].length = b.len;
820 	} else {
821 		switch (f->type) {
822 		case mmt_item:
823 		case mmt_berstring:
824 			if (a != 0) {
825 				if (at == fa_item) {
826 					val = getMappingItemVal(
827 						(__nis_mapping_item_t *)a,
828 						mit_any, rv,
829 		(f->type == mmt_berstring) ? f->match.berString : 0, NULL);
830 					if (numArg != 0)
831 						(*numArg)++;
832 				} else {
833 					val = cloneValue(
834 						(__nis_value_t *)a, 1);
835 					if (numArg != 0)
836 						(*numArg)++;
837 				}
838 			}
839 			break;
840 		case mmt_string:
841 			val = am(myself, sizeof (*val));
842 			if (val == 0)
843 				return (0);
844 			val->type = vt_string;
845 			val->numVals = 1;
846 			val->val = am(myself, sizeof (val->val[0]));
847 			if (val->val == 0) {
848 				sfree(val);
849 				return (0);
850 			}
851 			val->val[0].value = sdup(myself, T, f->match.string);
852 			val->val[0].length = strlen(val->val[0].value);
853 			break;
854 		case mmt_single:
855 		case mmt_limit:
856 		case mmt_any:
857 		case mmt_begin:
858 		case mmt_end:
859 			/* Not an error, so return an empty value */
860 			val = am(myself, sizeof (*val));
861 			if (val == 0)
862 				return (0);
863 			val->type = vt_string;
864 			val->numVals = 0;
865 			val->val = 0;
866 			break;
867 		default:
868 			/* Do nothing */
869 			val = 0;
870 			break;
871 		}
872 	}
873 	return (val);
874 }
875 
876 /*
877  * Used when evaluating an expression. Typically, the value of the
878  * expression so far will be kept in 'v1', and 'v2' is the value
879  * of the current component of the expression. In the general case,
880  * both will be multi-valued, and the result is an "explosion"
881  * resulting in N*M new values (if 'v1' had N values, and 'v2'
882  * M ditto).
883  *
884  * For example, if v1 = {"ab", "cd", "ef"}, and v2 = {"gh", "ij", "kl"},
885  * the result will be {"abgh", "abij", "abkl", "cdgh", "cdij", "cdkl",
886  * "efgh", "efij", "efkl"}.
887  *
888  * There are special cases when v1->repeat and/or v2->repeat are set.
889  * Repeat mostly makes sense with single values; for example, if
890  * v1 = {"x="} with repeat on, and v2 = {"1", "2", "3"}, the result
891  * is {"x=1", "x=2", "x=3"}.
892  *
893  * The result if v2 also had repeat on would be {"x=1x=2x=3"}. It's
894  * not clear if there's a useful application for this, but the code's
895  * there for the sake of orthogonality.
896  */
897 __nis_value_t *
898 explodeValues(__nis_value_t *v1, __nis_value_t *v2) {
899 	int		i1, i2, n, nv;
900 	__nis_value_t	*v;
901 	__nis_buffer_t	b = {0, 0};
902 	char		*myself = "explodeValues";
903 
904 	if (v1 == 0 || v1->numVals <= 0)
905 		return (cloneValue(v2, 1));
906 	if (v2 == 0 || v2->numVals <= 0)
907 		return (cloneValue(v1, 1));
908 
909 	/*
910 	 * XXX What should we do if (v1->type != v2->type) ?
911 	 * Policy: Just explode anyway, even though the result is
912 	 * unlikely to be very useful.
913 	 */
914 
915 	v = am(myself, sizeof (*v));
916 	if (v == 0)
917 		return (0);
918 
919 	if (!v1->repeat && !v2->repeat)
920 		nv = v1->numVals * v2->numVals;
921 	else if (v1->repeat && !v2->repeat)
922 		nv = v2->numVals;
923 	else if (!v1->repeat && v2->repeat)
924 		nv = v1->numVals;
925 	else /* v1->repeat && v2->repeat */
926 		nv = 1;
927 
928 	v->val = am(myself, nv * sizeof (v->val[0]));
929 	if (v->val == 0) {
930 		free(v);
931 		return (0);
932 	}
933 
934 	/*
935 	 * Four different cases, depending on the 'repeat' flags.
936 	 */
937 	if (!v1->repeat && !v2->repeat) {
938 		for (i1 = 0, n = 0; i1 < v1->numVals; i1++) {
939 			for (i2 = 0; i2 < v2->numVals; i2++) {
940 				if (v1->type == vt_string)
941 					sbc2buf(myself, v1->val[i1].value,
942 						v1->val[i1].length,
943 						&b);
944 				else
945 					bc2buf(myself, v1->val[i1].value,
946 						v1->val[i1].length,
947 						&b);
948 				if (v2->type == vt_string)
949 					sbc2buf(myself, v2->val[i2].value,
950 						v2->val[i2].length,
951 						&b);
952 				else
953 					bc2buf(myself, v2->val[i2].value,
954 						v2->val[i2].length,
955 						&b);
956 				v->val[n].value = b.buf;
957 				v->val[n].length = b.len;
958 				n++;
959 				b.buf = 0;
960 				b.len = 0;
961 			}
962 		}
963 	} else if (v1->repeat && !v2->repeat) {
964 		for (i2 = 0; i2 < v2->numVals; i2++) {
965 			for (i1 = 0, n = 0; i1 < v1->numVals; i1++) {
966 				if (v1->type == vt_string)
967 					sbc2buf(myself, v1->val[i1].value,
968 						v1->val[i1].length,
969 						&b);
970 				else
971 					bc2buf(myself, v1->val[i1].value,
972 						v1->val[i1].length,
973 						&b);
974 				if (v2->type == vt_string)
975 					sbc2buf(myself, v2->val[i2].value,
976 						v2->val[i2].length,
977 						&b);
978 				else
979 					bc2buf(myself, v2->val[i2].value,
980 						v2->val[i2].length,
981 						&b);
982 			}
983 			v->val[n].value = b.buf;
984 			v->val[n].length = b.len;
985 			n++;
986 			b.buf = 0;
987 			b.len = 0;
988 		}
989 	} else if (!v1->repeat && v2->repeat) {
990 		for (i1 = 0, n = 0; i1 < v1->numVals; i1++) {
991 			for (i2 = 0; i2 < v2->numVals; i2++) {
992 				if (v1->type == vt_string)
993 					sbc2buf(myself, v1->val[i1].value,
994 						v1->val[i1].length,
995 						&b);
996 				else
997 					bc2buf(myself, v1->val[i1].value,
998 						v1->val[i1].length,
999 						&b);
1000 				if (v2->type == vt_string)
1001 					sbc2buf(myself, v2->val[i2].value,
1002 						v2->val[i2].length,
1003 						&b);
1004 				else
1005 					bc2buf(myself, v2->val[i2].value,
1006 						v2->val[i2].length,
1007 						&b);
1008 			}
1009 			v->val[n].value = b.buf;
1010 			v->val[n].length = b.len;
1011 			n++;
1012 			b.buf = 0;
1013 			b.len = 0;
1014 		}
1015 	} else { /* v1->repeat && v2->repeat */
1016 		for (i1 = 0, n = 0; i1 < v1->numVals; i1++) {
1017 			for (i2 = 0; i2 < v2->numVals; i2++) {
1018 				if (v1->type == vt_string)
1019 					sbc2buf(myself, v1->val[i1].value,
1020 						v1->val[i1].length,
1021 						&b);
1022 				else
1023 					bc2buf(myself, v1->val[i1].value,
1024 						v1->val[i1].length,
1025 						&b);
1026 				if (v2->type == vt_string)
1027 					sbc2buf(myself, v2->val[i2].value,
1028 						v2->val[i2].length,
1029 						&b);
1030 				else
1031 					bc2buf(myself, v2->val[i2].value,
1032 						v2->val[i2].length,
1033 						&b);
1034 			}
1035 		}
1036 		v->val[n].value = b.buf;
1037 		v->val[n].length = b.len;
1038 		n++;
1039 		b.buf = 0;
1040 		b.len = 0;
1041 	}
1042 
1043 #ifdef	NISDB_LDAP_DEBUG
1044 	/* Sanity check */
1045 	if (n != nv)
1046 		abort();
1047 #endif	/* NISD__LDAP_DEBUG */
1048 
1049 	v->type = (v1->type == vt_string) ?
1050 			((v2->type == vt_string) ?
1051 				vt_string : vt_ber) : vt_ber;
1052 	v->repeat = 0;
1053 	v->numVals = n;
1054 
1055 	return (v);
1056 }
1057 
1058 __nis_value_t *
1059 getMappingFormatArray(__nis_mapping_format_t *a, __nis_rule_value_t *rv,
1060 			__nis_format_arg_t at, int numArgs, void *arg) {
1061 	int			i, ia = 0;
1062 	__nis_value_t		*val, *v = 0;
1063 	bool_t			moreFormat = (a != 0);
1064 	bool_t			moreArgs = (numArgs > 0);
1065 
1066 	while (moreFormat && (arg == 0 || ia < numArgs)) {
1067 		for (i = 0; moreFormat; i++) {
1068 			moreFormat = (a[i].type != mmt_end);
1069 			if (at == fa_item) {
1070 				__nis_mapping_item_t *item = arg;
1071 				val = getMappingFormat(&a[i], rv, at,
1072 					((item != 0) ? &item[ia] : 0), &ia);
1073 			} else {
1074 				__nis_value_t **ival = arg;
1075 				val = getMappingFormat(&a[i], rv, at,
1076 					((ival != 0) ? ival[ia] : 0), &ia);
1077 			}
1078 			if (val != 0) {
1079 				__nis_value_t	*new = explodeValues(v, val);
1080 
1081 				freeValue(v, 1);
1082 				freeValue(val, 1);
1083 				if (new == 0)
1084 					return (0);
1085 
1086 				v = new;
1087 			} else {
1088 				freeValue(v, 1);
1089 				return (0);
1090 			}
1091 			/*
1092 			 * If we run out of arguments, but still have format
1093 			 * remaining, repeat the last argument. Keep track of
1094 			 * the fact that we've really consumed all arguments.
1095 			 */
1096 			if (moreFormat && ia >= numArgs) {
1097 				ia = (numArgs > 0) ? numArgs - 1 : 0;
1098 				moreArgs = FALSE;
1099 			}
1100 		}
1101 		/*
1102 		 * We've run out of format, so if we still have arguments
1103 		 * left, start over on the format.
1104 		 */
1105 		if (ia < numArgs && moreArgs) {
1106 			/*
1107 			 * However, if we didn't consume any arguments going
1108 			 * through the format once, abort to avoid an infinite
1109 			 * loop.
1110 			 */
1111 			if (numArgs > 0 && ia <= 0) {
1112 				freeValue(v, 1);
1113 				return (0);
1114 			}
1115 			moreFormat = 1;
1116 		}
1117 	}
1118 
1119 	return (v);
1120 }
1121 
1122 /*
1123  * Returns a string representation (such as "[name=foo, value=bar]")
1124  * of a nis_index_t.
1125  */
1126 char *
1127 getIndex(__nis_index_t *i, int *len) {
1128 	int		n;
1129 	__nis_buffer_t	b = {0, 0};
1130 	char		*myself = "getIndex";
1131 
1132 	if (i == 0)
1133 		return (0);
1134 
1135 	if (i->numIndexes > 0) {
1136 		bp2buf(myself, &b, "[");
1137 		for (n = 0; n < i->numIndexes; n++) {
1138 			__nis_value_t	*val;
1139 			int		j;
1140 
1141 			val = getMappingFormatArray(i->value[n],
1142 						0, fa_any, 0, 0);
1143 			if (n > 0)
1144 				bp2buf(myself, &b, ", ");
1145 			bp2buf(myself, &b, "%s=", i->name[n]);
1146 			if (val != 0) {
1147 				for (j = 0; j < val->numVals; j++) {
1148 					bc2buf(myself, val->val[j].value,
1149 						val->val[j].length, &b);
1150 				}
1151 			} else {
1152 				bp2buf(myself, &b, "<no-vals>");
1153 			}
1154 			freeValue(val, 1);
1155 		}
1156 		bp2buf(myself, &b, "]");
1157 	}
1158 	if (len != 0)
1159 		*len = b.len;
1160 	return (b.buf);
1161 }
1162 
1163 char *
1164 getObjSpec(__nis_obj_spec_t *o, int *len) {
1165 	__nis_buffer_t	b = {0, 0};
1166 	char		*myself = "getObjSpec";
1167 
1168 	if (o == 0)
1169 		return (0);
1170 
1171 	b.buf = getIndex(&o->index, &b.len);
1172 	sbc2buf(myself, o->name, slen(o->name), &b);
1173 	if (len != 0)
1174 		*len = b.len;
1175 	return (b.buf);
1176 }
1177 
1178 /*
1179  * Returns a string representation of the LDAP scope. Note that the
1180  * returned value is a static entity, and must be copied by the
1181  * caller (but, obviously, must not be freed).
1182  */
1183 char *
1184 getScope(int scope) {
1185 	switch (scope) {
1186 	case LDAP_SCOPE_BASE:
1187 		return ("base");
1188 	case LDAP_SCOPE_ONELEVEL:
1189 		return ("one");
1190 	case LDAP_SCOPE_SUBTREE:
1191 		return ("sub");
1192 	default:
1193 		return ("one");
1194 	}
1195 }
1196 
1197 /*
1198  * Return a string representation of an LDAP search triple (such as
1199  * "ou=Hosts,dc=eng,dc=sun,dc=com?one?cn=xyzzy").
1200  */
1201 char *
1202 getSearchTriple(__nis_search_triple_t *s, int *len) {
1203 	__nis_buffer_t	b = {0, 0};
1204 	char		*a;
1205 	int		l;
1206 	char		*myself = "getSearchTriple";
1207 
1208 	/* If the scope is LDAP_SCOPE_UNKNOWN, the search triple is unused */
1209 	if (s == 0 || s->scope == LDAP_SCOPE_UNKNOWN) {
1210 		if (len != 0)
1211 			*len = 0;
1212 		return (0);
1213 	}
1214 
1215 	if (s->base != 0)
1216 		sbc2buf(myself, s->base, slen(s->base), &b);
1217 	if (!(s->scope == LDAP_SCOPE_ONELEVEL &&
1218 			(s->base == 0 || s->base[0] == '\0'))) {
1219 		bp2buf(myself, &b, "?%s?", getScope(s->scope));
1220 	}
1221 	if ((l = slen(s->attrs)) > 0) {
1222 		/*
1223 		 * Remove white space from the filter/attribute list.
1224 		 * The parser usually keeps any white space from the
1225 		 * config file (or LDAP/command line), but we don't
1226 		 * want it.
1227 		 */
1228 		a = am(myself, l+1);
1229 		if (a != 0) {
1230 			int	i, la;
1231 
1232 			for (i = 0, la = 0; i < l; i++) {
1233 				if (s->attrs[i] != ' ' &&
1234 						s->attrs[i] != '\t')
1235 					a[la++] = s->attrs[i];
1236 			}
1237 			sbc2buf(myself, a, la, &b);
1238 			sfree(a);
1239 		} else {
1240 			sbc2buf(myself, s->attrs, slen(s->attrs), &b);
1241 		}
1242 	}
1243 
1244 	if (len != 0)
1245 		*len = b.len;
1246 	return (b.buf);
1247 }
1248 
1249 __nis_value_t *
1250 getMappingItem(__nis_mapping_item_t *i, __nis_mapping_item_type_t native,
1251 		__nis_rule_value_t *rv, char *berstring, int *np_ldap_stat) {
1252 	char		*myself = "getMappingItem";
1253 	__nis_value_t	*val = 0;
1254 	__nis_buffer_t	b = {0, 0};
1255 	int		len = 0;
1256 	char		*buf;
1257 
1258 	if (i == 0)
1259 		return (0);
1260 
1261 	if (rv != 0)
1262 		return (getMappingItemVal(i, native, rv, berstring,
1263 			np_ldap_stat));
1264 
1265 	val = am(myself, sizeof (*val));
1266 	if (val == 0)
1267 		return (0);
1268 
1269 	switch (i->type) {
1270 	case mit_nisplus:
1271 		if (native != mit_nisplus)
1272 			bp2buf(myself, &b, "nis+:");
1273 		bp2buf(myself, &b, "%s", NIL(i->name));
1274 		buf = getObjSpec(&i->searchSpec.obj, &len);
1275 		if (buf != 0 && len > 0) {
1276 			bc2buf(myself, ":", 1, &b);
1277 			sbc2buf(myself, buf, len, &b);
1278 		}
1279 		sfree(buf);
1280 		val->type = vt_string;
1281 		val->repeat = i->repeat;
1282 		val->numVals = 1;
1283 		val->val = am(myself, sizeof (val->val[0]));
1284 		if (val->val == 0) {
1285 			sfree(b.buf);
1286 			free(val);
1287 			return (0);
1288 		}
1289 		val->val[0].value = b.buf;
1290 		val->val[0].length = b.len;
1291 		break;
1292 	case mit_ldap:
1293 		if (native != mit_ldap)
1294 			bp2buf(myself, &b, "ldap:");
1295 		bp2buf(myself, &b, "%s", NIL(i->name));
1296 		buf = getSearchTriple(&i->searchSpec.triple, &len);
1297 		if (buf != 0 && len > 0) {
1298 			bc2buf(myself, ":", 1, &b);
1299 			sbc2buf(myself, buf, len, &b);
1300 		}
1301 		sfree(buf);
1302 		val->type = vt_string;
1303 		val->repeat = i->repeat;
1304 		val->numVals = 1;
1305 		val->val = am(myself, sizeof (val->val[0]));
1306 		if (val->val == 0) {
1307 			sfree(b.buf);
1308 			free(val);
1309 			return (0);
1310 		}
1311 		val->val[0].value = b.buf;
1312 		val->val[0].length = b.len;
1313 		break;
1314 	default:
1315 		p2buf(myself, "<unknown>:");
1316 		p2buf(myself, "%s", NIL(i->name));
1317 		break;
1318 	}
1319 
1320 	return (val);
1321 }
1322 
1323 void
1324 copyObjSpec(__nis_obj_spec_t *old, __nis_obj_spec_t *new, int *err) {
1325 	char	*myself = "copyObjSpec";
1326 
1327 	if (old == 0 || new == 0) {
1328 		*err = EINVAL;
1329 		return;
1330 	}
1331 
1332 	if (new->index.name == 0) {
1333 		new->index.name = am(myself, old->index.numIndexes *
1334 						sizeof (new->index.name[0]));
1335 		if (old->index.numIndexes > 0 && new->index.name == 0) {
1336 			*err = ENOMEM;
1337 			return;
1338 		}
1339 		new->index.value = am(myself, old->index.numIndexes *
1340 						sizeof (new->index.value[0]));
1341 		if (old->index.numIndexes > 0 && new->index.value == 0) {
1342 			*err = ENOMEM;
1343 			return;
1344 		}
1345 	}
1346 	new->name = sdup(myself, T, old->name);
1347 	if (new->name == 0 && old->name != 0) {
1348 		*err = ENOMEM;
1349 		return;
1350 	}
1351 	copyIndex(&old->index, &new->index, err);
1352 }
1353 
1354 __nis_obj_spec_t *
1355 cloneObjSpec(__nis_obj_spec_t *old) {
1356 	char			*myself = "cloneObjSpec";
1357 	int			err = 0;
1358 	__nis_obj_spec_t	*new = am(myself, sizeof (*new));
1359 
1360 	if (new != 0) {
1361 		copyObjSpec(old, new, &err);
1362 		if (err != 0) {
1363 			freeObjSpec(new, 1);
1364 			new = 0;
1365 		}
1366 	}
1367 
1368 	return (new);
1369 }
1370 
1371 void
1372 freeObjSpec(__nis_obj_spec_t *old, bool_t doFree) {
1373 
1374 	if (old == 0)
1375 		return;
1376 
1377 	sfree(old->name);
1378 	freeIndex(&old->index, FALSE);
1379 	if (doFree)
1380 		free(old);
1381 }
1382 
1383 void
1384 copySearchTriple(__nis_search_triple_t *old, __nis_search_triple_t *new,
1385 		int *err) {
1386 	char			*myself = "copySearchTriple";
1387 
1388 	*err = 0;
1389 
1390 	if (old == 0 || new == 0) {
1391 		*err = EINVAL;
1392 		return;
1393 	}
1394 
1395 	if (old->base != NULL)
1396 		new->base = sdup(myself, T, old->base);
1397 	else
1398 		new->base = NULL;
1399 	if (old->attrs != NULL)
1400 		new->attrs = sdup(myself, T, old->attrs);
1401 	else
1402 		new->attrs = NULL;
1403 	if ((new->base == 0 && old->base != 0) ||
1404 			(new->attrs == 0 && old->attrs != 0)) {
1405 		sfree(new->base);
1406 		new->base = 0;
1407 		sfree(new->attrs);
1408 		new->attrs = 0;
1409 		*err = ENOMEM;
1410 		return;
1411 	}
1412 	new->scope = old->scope;
1413 	/*
1414 	 * XXX Really should have a cloneMappingElement() function.
1415 	 * However, since whatever the 'element' field points to
1416 	 * is allocated at parse time, and never is freed or modified,
1417 	 * it's sufficient to copy the pointer value.
1418 	 */
1419 	new->element = old->element;
1420 }
1421 
1422 __nis_search_triple_t *
1423 cloneSearchTriple(__nis_search_triple_t *old) {
1424 	char			*myself = "cloneSearchTriple";
1425 	int			err = 0;
1426 	__nis_search_triple_t	*new = am(myself, sizeof (*new));
1427 
1428 	if (new != 0) {
1429 		copySearchTriple(old, new, &err);
1430 		if (err != 0) {
1431 			freeSearchTriple(new, 1);
1432 			new = 0;
1433 		}
1434 	}
1435 
1436 	return (new);
1437 }
1438 
1439 void
1440 freeSearchTriple(__nis_search_triple_t *old, bool_t doFree) {
1441 
1442 	if (old == 0)
1443 		return;
1444 
1445 	sfree(old->base);
1446 	sfree(old->attrs);
1447 	/*
1448 	 * Since we only copied the element pointer when this structure
1449 	 * was created, we don't free old->element.
1450 	 */
1451 	if (doFree)
1452 		free(old);
1453 }
1454 
1455 void
1456 copyTripleOrObj(__nis_mapping_item_type_t type,
1457 		__nis_triple_or_obj_t *old, __nis_triple_or_obj_t *new,
1458 		int *err) {
1459 
1460 	*err = 0;
1461 
1462 	if (old == 0 || new == 0) {
1463 		*err = EINVAL;
1464 		return;
1465 	}
1466 
1467 	if (type == mit_nisplus) {
1468 		copyObjSpec(&old->obj, &new->obj, err);
1469 	} else if (type == mit_ldap) {
1470 		copySearchTriple(&old->triple, &new->triple, err);
1471 	}
1472 }
1473 
1474 __nis_triple_or_obj_t *
1475 cloneTripleOrObj(__nis_mapping_item_type_t type, __nis_triple_or_obj_t *old) {
1476 	char			*myself = "cloneTripleOrObj";
1477 	int			err = 0;
1478 	__nis_triple_or_obj_t	*new = am(myself, sizeof (*new));
1479 
1480 	if (new != 0) {
1481 		copyTripleOrObj(type, old, new, &err);
1482 		if (err != 0) {
1483 			freeTripleOrObj(type, new, 1);
1484 			new = 0;
1485 		}
1486 	}
1487 
1488 	return (new);
1489 }
1490 
1491 void
1492 freeTripleOrObj(__nis_mapping_item_type_t type, __nis_triple_or_obj_t *old,
1493 		bool_t doFree) {
1494 
1495 	if (old == 0)
1496 		return;
1497 
1498 	if (type == mit_nisplus)
1499 		freeObjSpec(&old->obj, doFree);
1500 	else if (type == mit_ldap)
1501 		freeSearchTriple(&old->triple, doFree);
1502 
1503 	if (doFree)
1504 		free(old);
1505 }
1506 
1507 void
1508 copyItem(__nis_mapping_item_t *old, __nis_mapping_item_t *new, int *err) {
1509 
1510 	*err = 0;
1511 
1512 	if (old == 0 || new == 0) {
1513 		*err = EINVAL;
1514 		return;
1515 	}
1516 
1517 	new->type = old->type;
1518 	new->repeat = old->repeat;
1519 	if (old->name != 0) {
1520 		new->name = strdup(old->name);
1521 		if (new->name == 0) {
1522 			*err = ENOMEM;
1523 			return;
1524 		}
1525 	} else {
1526 		new->name = 0;
1527 	}
1528 	if (old->type == mit_nisplus || old->type == mit_ldap)
1529 		copyTripleOrObj(old->type, &old->searchSpec, &new->searchSpec,
1530 				err);
1531 	else
1532 		memset(&new->searchSpec, 0, sizeof (new->searchSpec));
1533 }
1534 
1535 __nis_mapping_item_t *
1536 cloneItem(__nis_mapping_item_t *old) {
1537 	__nis_mapping_item_t	*new;
1538 	int			err = 0;
1539 	char			*myself = "cloneItem";
1540 
1541 	if (old == 0)
1542 		return (0);
1543 
1544 	new = am(myself, sizeof (*new));
1545 	if (new == 0)
1546 		return (0);
1547 
1548 	copyItem(old, new, &err);
1549 	if (err != 0) {
1550 		freeMappingItem(new, 1);
1551 		return (0);
1552 	}
1553 
1554 	return (new);
1555 }
1556 
1557 void
1558 freeMappingItem(__nis_mapping_item_t *item, int numItems) {
1559 	int	i;
1560 
1561 	if (item == 0)
1562 		return;
1563 
1564 	for (i = 0; i < numItems; i++) {
1565 		sfree(item[i].name);
1566 		freeTripleOrObj(item[i].type, &item[i].searchSpec, FALSE);
1567 	}
1568 	sfree(item);
1569 }
1570 
1571 __nis_mapping_item_t *
1572 concatenateMappingItem(__nis_mapping_item_t *old, int numItems,
1573 		__nis_mapping_item_t *cat) {
1574 	__nis_mapping_item_t	*new;
1575 	int			i, err = 0;
1576 	char			*myself = "concatenateMappingItem";
1577 
1578 	if (old == 0 || numItems < 1)
1579 		return (cloneItem(cat));
1580 
1581 	new = am(myself, (numItems + 1) * sizeof (*new));
1582 	if (new == 0)
1583 		return (0);
1584 
1585 	for (i = 0; i < numItems; i++) {
1586 		copyItem(&old[i], &new[i], &err);
1587 		if (err != 0) {
1588 			freeMappingItem(new, i);
1589 			return (0);
1590 		}
1591 	}
1592 	copyItem(cat, &new[numItems], &err);
1593 	if (err != 0) {
1594 		freeMappingItem(new, numItems);
1595 		new = 0;
1596 	}
1597 
1598 	return (new);
1599 }
1600 
1601 __nis_value_t *
1602 concatenateValues(__nis_value_t *v1, __nis_value_t *v2) {
1603 	int		i, n, a;
1604 	__nis_value_t	*v;
1605 	char		*myself = "concatenateValues";
1606 
1607 	if (v1 == 0 || v1->numVals <= 0)
1608 		return (cloneValue(v2, 1));
1609 	if (v2 == 0 || v2->numVals <= 0)
1610 		return (cloneValue(v1, 1));
1611 
1612 	if (v1->type != v2->type)
1613 		return (0);
1614 
1615 	n = v1->numVals + v2->numVals;
1616 	v = am(myself, sizeof (*v));
1617 	if (v == 0)
1618 		return (0);
1619 	v->val = am(myself, n * sizeof (v->val[0]));
1620 	if (v->val == 0) {
1621 		free(v);
1622 		return (0);
1623 	}
1624 	v->type = v1->type;
1625 	v->numVals = 0;
1626 
1627 	for (a = 0; a < 2; a++) {
1628 		__nis_single_value_t	*val = (a == 0) ? v1->val : v2->val;
1629 		int			numv = (a == 0) ? v1->numVals :
1630 							v2->numVals;
1631 		for (i = 0; i < numv; i++) {
1632 			int	clen, alen = val[i].length;
1633 
1634 			clen = alen;
1635 
1636 			/*
1637 			 * Make sure there's a NUL at the end of a string,
1638 			 * but avoid adding to the allocated length if there's
1639 			 * already a NUL at the end.
1640 			 */
1641 			if (alen > 0 && v->type == vt_string &&
1642 					((char *)val[i].value)[alen-1] != '\0')
1643 				alen += 1;
1644 			v->val[v->numVals].value = am(myself, alen);
1645 			if (v->val[v->numVals].value == 0) {
1646 				freeValue(v, 1);
1647 				return (0);
1648 			}
1649 			memcpy(v->val[v->numVals].value, val[i].value, clen);
1650 			v->val[v->numVals].length = val[i].length;
1651 			v->numVals++;
1652 		}
1653 	}
1654 
1655 	return (v);
1656 }
1657 
1658 __nis_value_t *
1659 splitMappingItem(__nis_mapping_item_t *item, char delim,
1660 		__nis_rule_value_t *rv) {
1661 	__nis_value_t		*val = getMappingItem(item, mit_any,
1662 			rv, 0, NULL);
1663 	__nis_single_value_t	*nval;
1664 	int			i, n, nv;
1665 
1666 	if (val == 0)
1667 		return (0);
1668 	else if (delim == 0 || val->val == 0 || val->numVals <= 0 ||
1669 			val->type != vt_string) {
1670 		freeValue(val, 1);
1671 		return (0);
1672 	}
1673 
1674 	nval = val->val;
1675 	nv = val->numVals;
1676 	val->repeat = FALSE;
1677 	val->val = 0;
1678 	val->numVals = 0;
1679 
1680 	/* In N2L, space and tab delimiters are treated the same */
1681 	if (yp2ldap && delim == '\t')
1682 		delim = ' ';
1683 
1684 	/* If the item has multiple values, we split each one independently */
1685 	for (i = 0; i < nv; i++) {
1686 		char			*str;
1687 		int			s, e;
1688 		char			*newstr;
1689 		__nis_single_value_t	*newval;
1690 
1691 		if (yp2ldap && delim == ' ')
1692 			nval[i].value = trimWhiteSpaces(nval[i].value,
1693 							&nval[i].length, 1);
1694 
1695 		str = nval[i].value;
1696 
1697 		if (nval[i].value == 0)
1698 			continue;
1699 
1700 		for (s = 0; s < nval[i].length; s = e+1) {
1701 			/* Find the next delimiter, or end-of-string */
1702 			for (e = s; str[e] != '\0' && str[e] != delim; e++);
1703 			/*
1704 			 * 'str[e]' is either a delimiter, or the concluding
1705 			 * NUL. Make sure it's NUL.
1706 			 */
1707 			str[e] = '\0';
1708 			/* Add to val->val */
1709 			newstr = strdup(&str[s]);
1710 			newval = realloc(val->val,
1711 					(val->numVals+1) *
1712 						sizeof (val->val[0]));
1713 			if (newval != 0)
1714 				val->val = newval;
1715 			if (newstr == 0 || newval == 0) {
1716 				freeValue(val, 1);
1717 				for (n = i; n < nv; n++) {
1718 					sfree(nval[n].value);
1719 				}
1720 				free(nval);
1721 				sfree(newstr);
1722 				return (0);
1723 			}
1724 			val->val[val->numVals].value = newstr;
1725 			val->val[val->numVals].length = strlen(newstr) + 1;
1726 			val->numVals++;
1727 		}
1728 		free(nval[i].value);
1729 		nval[i].value = 0;
1730 	}
1731 	/* Already freed the nval[i].value's as we traversed nval */
1732 	free(nval);
1733 
1734 	return (val);
1735 }
1736 
1737 /*
1738  * Match the format spec 'f[curf]' to the input value string 'str'.
1739  *
1740  * If successful, returns the updated position in the value string 'str'.
1741  * Otherwise, NULL is returned.
1742  *
1743  * curf		Current index (i.e., the one we should look at) in 'f'
1744  * nf		Number of elements in 'f', including 'mmt_end'
1745  * str		The value string we're scanning
1746  * val		Pointer to where an item value (if any) should be returned
1747  *		Set to NULL if not an 'mmt_item'.
1748  * fmtstart	If non-zero on entry, skip characters in 'str' until we find
1749  *		the f[curf].type data, if doing so makes any sense. On exit,
1750  *		set to the start of the fmt element data (which will be 'str',
1751  *		unless we did skip characters)
1752  * sepset	List of separators
1753  */
1754 char *
1755 scanMappingFormat(__nis_mapping_format_t *f, int curf, int nf, char *str,
1756 		char **val, char **fmtstart, char *sepset) {
1757 	char	*mstr, *next, *start = 0, *tmpstr;
1758 	int	i, len;
1759 	bool_t	match;
1760 	char	*myself = "scanMappingFormat";
1761 	/* N2L variables */
1762 	int	af, skipspaces = 0;
1763 	bool_t	ipaddr = FALSE;
1764 	char	*spacestr = " ", *emptystr = "";
1765 
1766 
1767 	if (f == 0 || curf < 0 || nf <= 0 || str == 0)
1768 		return (0);
1769 
1770 	/*
1771 	 * If separator list is NULL (which will be the case for
1772 	 * nis+2ldap), then simply use empty string
1773 	 */
1774 	if (sepset == 0)
1775 		sepset = emptystr;
1776 
1777 	if (curf >= nf) {
1778 		/* OK if the string also is exhausted */
1779 		if (strchr(sepset, *str) != 0)
1780 			return (str);
1781 		else
1782 			return (0);
1783 	}
1784 
1785 	switch (f[curf].type) {
1786 	case mmt_berstring:
1787 		if (f[curf].match.berString[0] != 'a') {
1788 			/* Not a matchable element */
1789 			return (0);
1790 		}
1791 
1792 		/*
1793 		 * If here, it means it's an IP address (N2L case)
1794 		 * So continue processing as if it was mmt_item
1795 		 */
1796 		ipaddr = TRUE;
1797 
1798 	case mmt_item:
1799 		/*
1800 		 * In order to find the end of the item value, we must look
1801 		 * ahead and determine the start of the next formatting element.
1802 		 * If successful, 'next' will be the start of the fmt element
1803 		 * after the next one; we don't care about that, other than to
1804 		 * check for error.
1805 		 *
1806 		 * Since an item match is somewhat like an any match, in that
1807 		 * we don't know a priori if the first occurence of the next
1808 		 * element really is the one we want, we have to scan ahead
1809 		 * until we've reached the end.
1810 		 */
1811 		tmpstr = str;
1812 		while ((next = scanMappingFormat(f, curf+1, nf, tmpstr, 0,
1813 				&start, sepset)) != 0) {
1814 			char	*tmp = next;
1815 			int	cf;
1816 
1817 			for (cf = curf+2; cf < nf; cf++) {
1818 				tmp = scanMappingFormat(f, cf, nf, tmp, 0,
1819 					0, sepset);
1820 				if (tmp == 0)
1821 					break;
1822 			}
1823 			if (tmp == 0) {
1824 				tmpstr = next;
1825 			} else if (strchr(sepset, *tmp) != 0) {
1826 				break;
1827 			} else {
1828 				return (0);
1829 			}
1830 
1831 		}
1832 		if (next == 0 || start == 0)
1833 			return (0);
1834 
1835 		if (val != 0) {
1836 			len = (int)((long)start - (long)str);
1837 			*val = am(myself, len + 1);
1838 			if (*val == 0)
1839 				return (0);
1840 			memcpy(*val, str, len);
1841 			(*val)[len] = '\0';
1842 
1843 			if (ipaddr == TRUE) {
1844 				/*
1845 				 * In N2L, we need to check if *val is truly an
1846 				 * IP address
1847 				 */
1848 				af = checkIPaddress(*val, len, &tmpstr);
1849 
1850 				if (af == -2) {
1851 					logmsg(MSG_NOTIMECHECK, LOG_WARNING,
1852 						"%s:Internal error while "
1853 						"processing IPaddress %s",
1854 						myself, *val);
1855 					sfree(*val);
1856 					return (0);
1857 				} else if (af == -1) {
1858 					logmsg(MSG_NOTIMECHECK, LOG_WARNING,
1859 						"%s:%s is not an IP address",
1860 						myself, *val);
1861 					sfree(*val);
1862 					return (0);
1863 				} else if (af == 0) {
1864 					logmsg(MSG_NOTIMECHECK, LOG_WARNING,
1865 						"%s:IP address %s is not "
1866 						"supported by rfc2307bis",
1867 						myself, *val);
1868 					sfree(*val);
1869 					return (0);
1870 				} else if (sstrncmp(*val, tmpstr, len) != 0) {
1871 					logmsg(MSG_NOTIMECHECK, LOG_WARNING,
1872 						"%s:IPaddress %s converted "
1873 						"to %s", myself, *val, tmpstr);
1874 				}
1875 
1876 				sfree(*val);
1877 				*val = tmpstr;
1878 			}
1879 		}
1880 
1881 		if (fmtstart != 0)
1882 			*fmtstart = str;
1883 
1884 		return (start);
1885 
1886 	case mmt_string:
1887 		if ((mstr = f[curf].match.string) == 0 || *mstr == '\0') {
1888 			/*
1889 			 * Count this as a successful match of an empty
1890 			 * string.
1891 			 */
1892 			if (fmtstart != 0)
1893 				*fmtstart = str;
1894 			return (str);
1895 		}
1896 
1897 		/*
1898 		 * In N2L, if the format string 'mstr' contains only
1899 		 * whitespaces (spaces and tabs), then it should
1900 		 * match one or more whitespaces from the input
1901 		 * string 'str'.
1902 		 */
1903 		if (yp2ldap && strspn(mstr, " \t") == strlen(mstr)) {
1904 				mstr = spacestr;
1905 				skipspaces = 1;
1906 				next = str + strcspn(str, " \t");
1907 				/*
1908 				 * Even if there is no whitespace in 'str',
1909 				 * it's OK. This is to allow formats like
1910 				 * "%s %s %s" to match inputs like "foo bar".
1911 				 */
1912 				if (*next == '\0')
1913 					mstr = emptystr;
1914 		} else {
1915 			/* No match string in 'str' => failure */
1916 			if ((next = strstr(str, mstr)) == 0)
1917 				return (0);
1918 		}
1919 
1920 		/* If 'fmtstart' == 0, we require 'next' == 'str' */
1921 		if (fmtstart == 0 && next != str)
1922 			return (0);
1923 		/* Success; save start of match string if requested */
1924 		if (fmtstart != 0)
1925 			*fmtstart = next;
1926 		/* Update position in the value string */
1927 		str = (char *)((long)next + (long)strlen(mstr));
1928 
1929 		/* Skip whitespaces for N2L */
1930 		if (skipspaces == 1)
1931 			for (; *str == ' ' || *str == '\t'; str++);
1932 
1933 		return (str);
1934 
1935 	case mmt_single:
1936 		if (fmtstart != 0) {
1937 			match = FALSE;
1938 			/* Skip ahead until we match */
1939 			for (next = str; *next != '\0'; next++) {
1940 				unsigned char	*lo = f[curf].match.single.lo;
1941 				unsigned char	*hi = f[curf].match.single.hi;
1942 
1943 				for (i = 0; i < f[curf].match.single.numRange;
1944 						i++) {
1945 					if (*next >= lo[i] && *next <= hi[i]) {
1946 						match = TRUE;
1947 						break;
1948 					}
1949 				}
1950 				if (match)
1951 					break;
1952 			}
1953 			if (!match)
1954 				return (0);
1955 			*fmtstart = next;
1956 			str = next;
1957 		} else {
1958 			match = FALSE;
1959 			for (i = 0; i < f[curf].match.single.numRange; i++) {
1960 				if (*str >= f[curf].match.single.lo[i] &&
1961 					*str <= f[curf].match.single.hi[i]) {
1962 					match = TRUE;
1963 					break;
1964 				}
1965 			}
1966 			if (!match)
1967 				return (0);
1968 		}
1969 		/* Step over the matched character */
1970 		str++;
1971 		return (str);
1972 
1973 	case mmt_any:
1974 		/*
1975 		 * Look ahead to find the beginning of the next element.
1976 		 * Because a wildcard-match isn't necessarily uniquely
1977 		 * determined until we've reached the end, we then continue
1978 		 * to scan ahead.
1979 		 */
1980 		while ((next = scanMappingFormat(f, curf+1, nf, str, 0,
1981 						&start, sepset)) != 0) {
1982 			char	*tmp = next;
1983 			int	cf;
1984 
1985 			for (cf = curf+2; cf < nf; cf++) {
1986 				tmp = scanMappingFormat(f, cf, nf, tmp, 0,
1987 					0, sepset);
1988 				if (tmp == 0)
1989 					break;
1990 			}
1991 			if (tmp == 0) {
1992 				str = next;
1993 			} else if (*tmp == '\0') {
1994 				break;
1995 			} else {
1996 				return (0);
1997 			}
1998 		}
1999 		if (next == 0 || start == 0)
2000 			return (0);
2001 
2002 		if (fmtstart != 0)
2003 			*fmtstart = str;
2004 
2005 		return (start);
2006 
2007 	case mmt_limit:
2008 		if (f[curf].match.limit == eos) {
2009 			if (fmtstart != 0) {
2010 				/* Skip to the end */
2011 				str = str + strcspn(str, sepset);
2012 				*fmtstart = str;
2013 			} else if (strchr(sepset, *str) == 0) {
2014 				return (0);
2015 			}
2016 		}
2017 		return (str);
2018 
2019 	case mmt_begin:
2020 		if (fmtstart != 0)
2021 			*fmtstart = str;
2022 		return (str);
2023 
2024 	case mmt_end:
2025 		if (fmtstart != 0) {
2026 			/* Skip to the end */
2027 			str = str + strcspn(str, sepset);
2028 			*fmtstart = str;
2029 			return (str);
2030 		}
2031 		/* No skipping, so we must be at the end of the value */
2032 		if (strchr(sepset, *str) == 0)
2033 			return (0);
2034 		return (str);
2035 
2036 	default:
2037 		break;
2038 	}
2039 
2040 	return (0);
2041 }
2042 
2043 /*
2044  * Verify that the string 'str' matches the mapping format array 'f'.
2045  * Returns 1 in case of a match, 0 otherwise.
2046  */
2047 int
2048 verifyMappingMatch(__nis_mapping_format_t *f, char *str) {
2049 	int			n, nf;
2050 	__nis_mapping_format_t	*ftmp;
2051 
2052 	/* Count the number of format elements in the format */
2053 	for (nf = 0, ftmp = f; ftmp->type != mmt_end; ftmp++) {
2054 		nf++;
2055 	}
2056 	/* Count the mmt_end as well */
2057 	nf++;
2058 
2059 	for (n = 0; n < nf; n++) {
2060 		str = scanMappingFormat(f, n, nf, str, 0, 0, 0);
2061 		if (str == 0)
2062 			break;
2063 	}
2064 
2065 	return ((str != 0) ? 1 : 0);
2066 }
2067 
2068 /*
2069  * Perform a match operation. For example, given the rule
2070  *	("{%s}%s", auth_name, public_data)=nisPublicKey
2071  * and assuming that 'nisPublicKey' has the value "{dh640-0}abcdef12345",
2072  * assign "dh640-0" to 'auth_name' and "abcdef12345" to 'public_data'.
2073  *
2074  * Note that this function doesn't perform the actual assignment. Rather,
2075  * it returns an array of __nis_value_t's, with element zero of the value
2076  * array being the new value of the first matched item, element one the
2077  * value of the second matched item, etc. In the example above, we'd
2078  * return a value array with two elements.
2079  *
2080  * If there is more than one input value (inVal->numVals > 1), the
2081  * output array elements will also be multi-valued.
2082  *
2083  * f		The match format
2084  * inVal	Input value(s)
2085  * numVal	Number of elements in the output value array
2086  * sepset	List of separators
2087  * outstr	Points to the updated position upto which the
2088  *		input string has been matched
2089  */
2090 __nis_value_t **
2091 matchMappingItem(__nis_mapping_format_t *f, __nis_value_t *inVal,
2092 		int *numVals, char *sepset, char **outstr) {
2093 	__nis_value_t		**v = 0;
2094 	int			i, n, ni, numItems, nf;
2095 	char			*str, *valstr;
2096 	__nis_mapping_format_t	*ftmp;
2097 	char			*myself = "matchMappingItem";
2098 
2099 	if (f == 0 ||
2100 		inVal == 0 || inVal->numVals < 1 || inVal->type != vt_string)
2101 		return (0);
2102 
2103 	/* Count the number of format elements and items in the format */
2104 	for (nf = numItems = 0, ftmp = f; ftmp->type != mmt_end; ftmp++) {
2105 		nf++;
2106 
2107 		/*
2108 		 * Count mmt_item and mmt_berstring (used by N2L to
2109 		 * represent address %a)
2110 		 */
2111 		if (ftmp->type == mmt_item)
2112 				numItems++;
2113 		else if (ftmp->type == mmt_berstring && ftmp->match.berString &&
2114 				ftmp->match.berString[0] == 'a')
2115 				numItems++;
2116 	}
2117 	/* Count the mmt_end as well */
2118 	nf++;
2119 
2120 	/*
2121 	 * If no items, there will be no values. This isn't exactly an error
2122 	 * from the limited point of view of this function, so we return a
2123 	 * __nis_value_t with zero values.
2124 	 */
2125 	if (numItems <= 0) {
2126 		v = am(myself, sizeof (v[0]));
2127 		if (v == 0)
2128 			return (0);
2129 		v[0] = am(myself, sizeof (*v[0]));
2130 		if (v[0] == 0) {
2131 			sfree(v);
2132 			return (0);
2133 		}
2134 		v[0]->type = vt_string;
2135 		v[0]->numVals = 0;
2136 		v[0]->val = 0;
2137 		if (numVals != 0)
2138 			*numVals = 1;
2139 		return (v);
2140 	}
2141 
2142 	/* Allocate and initialize the return array */
2143 	v = am(myself, numItems * sizeof (v[0]));
2144 	if (v == 0)
2145 		return (0);
2146 	for (n = 0; n < numItems; n++) {
2147 		v[n] = am(myself, sizeof (*v[n]));
2148 		if (v[n] == 0) {
2149 			int	j;
2150 
2151 			for (j = 0; j < n; j++)
2152 				freeValue(v[j], 1);
2153 			sfree(v);
2154 			return (0);
2155 		}
2156 		v[n]->type = vt_string;
2157 		v[n]->numVals = 0;
2158 		v[n]->val = am(myself, inVal->numVals * sizeof (v[n]->val[0]));
2159 		if (v[n]->val == 0) {
2160 			int	j;
2161 
2162 			for (j = 0; j < n; j++)
2163 				freeValue(v[j], 1);
2164 			sfree(v);
2165 			return (0);
2166 		}
2167 		for (i = 0; i < inVal->numVals; i++) {
2168 			v[n]->val[i].length = 0;
2169 			v[n]->val[i].value = 0;
2170 		}
2171 	}
2172 
2173 	/* For each input value, perform the match operation */
2174 	for (i = 0; i < inVal->numVals; i++) {
2175 		str = inVal->val[i].value;
2176 		if (str == 0)
2177 			continue;
2178 		for (n = 0, ni = 0; n < nf; n++) {
2179 			valstr = 0;
2180 			str = scanMappingFormat(f, n, nf, str, &valstr,
2181 				0, sepset);
2182 			if (str == 0)
2183 				break;
2184 			if (valstr != 0 && ni < numItems &&
2185 					v[ni]->numVals < inVal->numVals) {
2186 				v[ni]->val[v[ni]->numVals].value = valstr;
2187 				v[ni]->val[v[ni]->numVals].length =
2188 							strlen(valstr) + 1;
2189 				v[ni]->numVals++;
2190 				ni++;
2191 			} else if (valstr != 0) {
2192 				sfree(valstr);
2193 			}
2194 		}
2195 		if (str == 0) {
2196 			for (n = 0; n < numItems; n++)
2197 				freeValue(v[n], 1);
2198 			sfree(v);
2199 			return (0);
2200 		}
2201 	}
2202 
2203 	if (numVals != 0)
2204 		*numVals = numItems;
2205 
2206 	/*
2207 	 * Update the return string upto the point it has been matched
2208 	 * This string will be used by the N2L code in its next call
2209 	 * to this function
2210 	 */
2211 	if (outstr != 0)
2212 		*outstr = str;
2213 
2214 	return (v);
2215 }
2216 
2217 /*
2218  * Perform an extract operation. For example, given the expression
2219  *	(name, "%s.*")
2220  * and assuming 'name' is an item with the value "some.thing", the
2221  * value returned by the extract is "some".
2222  */
2223 __nis_value_t *
2224 extractMappingItem(__nis_mapping_item_t *item, __nis_mapping_format_t *f,
2225 		__nis_rule_value_t *rv, int *stat) {
2226 	__nis_value_t		*val = getMappingItem(item, mit_any,
2227 			rv, 0, stat);
2228 	__nis_single_value_t	*nval;
2229 	int			i, n, nv, nf;
2230 	__nis_mapping_format_t	*ftmp;
2231 
2232 	if (val == 0)
2233 		return (0);
2234 	else if (f == 0 || rv == 0 || val->val == 0 ||
2235 			val->numVals <= 0 || val->type != vt_string) {
2236 		freeValue(val, 1);
2237 		return (0);
2238 	}
2239 
2240 	/* Sanity check the format; it must have one and only one mmt_item */
2241 	{
2242 		int	numitem;
2243 
2244 		for (nf = numitem = 0, ftmp = f; ftmp->type != mmt_end;
2245 				ftmp++) {
2246 			nf++;
2247 			if (ftmp->type == mmt_item)
2248 				numitem++;
2249 		}
2250 		/* Count the mmt_end as well */
2251 		nf++;
2252 		if (numitem != 1) {
2253 			freeValue(val, 1);
2254 			return (0);
2255 		}
2256 	}
2257 
2258 	nval = val->val;
2259 	nv = val->numVals;
2260 	val->repeat = FALSE;
2261 	val->val = 0;
2262 	val->numVals = 0;
2263 
2264 	/* If the item has multiple values, we extract each one independently */
2265 	for (i = 0; i < nv; i++) {
2266 		char			*str = nval[i].value;
2267 		char			*newstr = 0;
2268 		__nis_single_value_t	*newval;
2269 
2270 		if (nval[i].value == 0)
2271 			continue;
2272 
2273 		/*
2274 		 * We match the whole string, even if we find a value for
2275 		 * the item before exhausting all format elements. By doing
2276 		 * this, we ensure that the string really matches the complete
2277 		 * format specification.
2278 		 */
2279 		for (n = 0; n < nf; n++) {
2280 			str = scanMappingFormat(f, n, nf, str, &newstr, 0, 0);
2281 			if (str == 0)
2282 				break;
2283 		}
2284 
2285 		/*
2286 		 * *str should now be NUL, meaning we've reached the end of
2287 		 * the string (value), and it completely matched the format.
2288 		 * If 'str' is NULL, there was an error, and if 'newstr' is
2289 		 * 0, we somehow failed to obtain a value.
2290 		 */
2291 		if (str == 0 || *str != '\0' || newstr == 0 ||
2292 				(newval = realloc(val->val,
2293 					(val->numVals+1) *
2294 					sizeof (val->val[0]))) == 0) {
2295 			freeValue(val, 1);
2296 			for (n = 0; n < nv; n++) {
2297 				sfree(nval[n].value);
2298 			}
2299 			free(nval);
2300 			sfree(newstr);
2301 			return (0);
2302 		}
2303 
2304 		val->val = newval;
2305 		val->val[val->numVals].value = newstr;
2306 		val->val[val->numVals].length = strlen(newstr) + 1;
2307 		val->numVals++;
2308 
2309 		free(nval[i].value);
2310 		nval[i].value = 0;
2311 	}
2312 	free(nval);
2313 
2314 	return (val);
2315 }
2316 
2317 /*
2318  * For each value in 'val', remove the last character, provided that
2319  * it matches 'elide'.
2320  */
2321 void
2322 stringElide(__nis_value_t *val, char elide) {
2323 
2324 	if (val != 0 && val->type == vt_string) {
2325 		int	i;
2326 
2327 		for (i = 0; i < val->numVals; i++) {
2328 			int	end = val->val[i].length;
2329 			char	*str = val->val[i].value;
2330 
2331 			if (str == 0 || end <= 0)
2332 				continue;
2333 
2334 			/*
2335 			 * If the NUL was counted in the length, step back
2336 			 * over it.
2337 			 */
2338 			if (str[end-1] == '\0')
2339 				end--;
2340 			if (end > 0 && str[end-1] == elide) {
2341 				str[end-1] = '\0';
2342 				val->val[i].length--;
2343 			}
2344 		}
2345 	}
2346 }
2347 
2348 /*
2349  * Obtain the value for the mapping sub-element 'e', given the input
2350  * rule-value 'rv'.
2351  */
2352 __nis_value_t *
2353 getMappingSubElement(__nis_mapping_sub_element_t *e,
2354 	__nis_rule_value_t *rv, int *np_ldap_stat) {
2355 	__nis_value_t	*val;
2356 
2357 	if (e == 0)
2358 		return (0);
2359 
2360 	switch (e->type) {
2361 	case me_item:
2362 		val = getMappingItem(&e->element.item, mit_any, rv, 0,
2363 			np_ldap_stat);
2364 		break;
2365 	case me_print:
2366 		val = getMappingFormatArray(e->element.print.fmt, rv,
2367 						fa_item,
2368 						e->element.print.numItems,
2369 						e->element.print.item);
2370 		if (e->element.print.doElide)
2371 			stringElide(val, e->element.print.elide);
2372 		break;
2373 	case me_split:
2374 		val = splitMappingItem(&e->element.split.item,
2375 					e->element.split.delim,
2376 					rv);
2377 		break;
2378 	case me_extract:
2379 		val = extractMappingItem(&e->element.extract.item,
2380 					e->element.extract.fmt,
2381 					rv, np_ldap_stat);
2382 		break;
2383 	case me_match:
2384 	default:
2385 		val = 0;
2386 		break;
2387 	}
2388 
2389 	return (val);
2390 }
2391 
2392 /*
2393  * Obtain the value of the mapping element 'e', given the input rule-
2394  * value 'rv'. The 'native' mapping type is used when 'rv' is NULL,
2395  * and the result is a string representation of the mapping element;
2396  * in that case, items of the 'native' type are printed without their
2397  * type designation ("nis+" or "ldap").
2398  */
2399 __nis_value_t *
2400 getMappingElement(__nis_mapping_element_t *e, __nis_mapping_item_type_t native,
2401 		__nis_rule_value_t *rv, int *stat) {
2402 	__nis_value_t	*val, **tv;
2403 	int		i, success = 0, novalue = 0;
2404 	int *np_ldap_stat;
2405 	char		*myself = "getMappingElement";
2406 
2407 	switch (e->type) {
2408 	case me_item:
2409 		val = getMappingItem(&e->element.item, native, rv, 0, NULL);
2410 		break;
2411 	case me_print:
2412 		tv = am(myself, e->element.print.numSubElements *
2413 			sizeof (tv[0]));
2414 		np_ldap_stat = am(myself,
2415 			e->element.print.numSubElements * sizeof (int));
2416 		if ((e->element.print.numSubElements > 0) &&
2417 				(tv == 0 || np_ldap_stat == 0)) {
2418 			val = 0;
2419 			sfree(tv);
2420 			sfree(np_ldap_stat);
2421 			break;
2422 		}
2423 		for (i = 0; i < e->element.print.numSubElements; i++) {
2424 			np_ldap_stat[i] = 0;
2425 			tv[i] = getMappingSubElement(
2426 				&e->element.print.subElement[i],
2427 				rv, &np_ldap_stat[i]);
2428 		}
2429 		/*
2430 		 * if we get NP_LDAP_NO_VALUE to any of the subelement
2431 		 * and we get NP_LDAP_MAP_SUCCESS to all other subelement
2432 		 * then we had enough nis+ column values which can
2433 		 * produce value for this rule, but didn't. So return
2434 		 * NP_LDAP_RULES_NO_VALUE to indicate to proceed to
2435 		 * next database id.
2436 		 */
2437 		for (i = 0; i < e->element.print.numSubElements; i++) {
2438 			if (np_ldap_stat[i] == NP_LDAP_MAP_SUCCESS)
2439 				success++;
2440 			if (np_ldap_stat[i] == NP_LDAP_NO_VALUE)
2441 				novalue++;
2442 		}
2443 		if (stat != NULL && novalue > 0 &&
2444 				((novalue+success) ==
2445 					e->element.print.numSubElements))
2446 					    *stat = NP_LDAP_RULES_NO_VALUE;
2447 		val = getMappingFormatArray(e->element.print.fmt, rv,
2448 						fa_value,
2449 						e->element.print.numSubElements,
2450 						tv);
2451 		for (i = 0; i < e->element.print.numSubElements; i++) {
2452 			freeValue(tv[i], 1);
2453 		}
2454 		sfree(tv);
2455 		sfree(np_ldap_stat);
2456 		if (e->element.print.doElide)
2457 			stringElide(val, e->element.print.elide);
2458 		break;
2459 	case me_split:
2460 		val = splitMappingItem(&e->element.split.item,
2461 					e->element.split.delim,
2462 					rv);
2463 		break;
2464 	case me_match:
2465 		/*
2466 		 * A match doesn't produce an assignable value per se,
2467 		 * so we shouldn't get one here.
2468 		 */
2469 		val = 0;
2470 		break;
2471 	case me_extract:
2472 		val = extractMappingItem(&e->element.extract.item,
2473 					e->element.extract.fmt,
2474 					rv, NULL);
2475 		break;
2476 	default:
2477 		val = 0;
2478 		break;
2479 	}
2480 
2481 	return (val);
2482 }
2483