xref: /titanic_44/usr/src/lib/libldap5/sources/ldap/common/disptmpl.c (revision 004388ebfdfe2ed7dfd2d153a876dfcc22d2c006)
17c478bd9Sstevel@tonic-gate /*
2*004388ebScasper  * Copyright 2006 Sun Microsystems, Inc.  All rights reserved.
37c478bd9Sstevel@tonic-gate  * Use is subject to license terms.
47c478bd9Sstevel@tonic-gate  */
57c478bd9Sstevel@tonic-gate 
67c478bd9Sstevel@tonic-gate #pragma ident	"%Z%%M%	%I%	%E% SMI"
77c478bd9Sstevel@tonic-gate 
87c478bd9Sstevel@tonic-gate 
97c478bd9Sstevel@tonic-gate /*
107c478bd9Sstevel@tonic-gate  * The contents of this file are subject to the Netscape Public
117c478bd9Sstevel@tonic-gate  * License Version 1.1 (the "License"); you may not use this file
127c478bd9Sstevel@tonic-gate  * except in compliance with the License. You may obtain a copy of
137c478bd9Sstevel@tonic-gate  * the License at http://www.mozilla.org/NPL/
147c478bd9Sstevel@tonic-gate  *
157c478bd9Sstevel@tonic-gate  * Software distributed under the License is distributed on an "AS
167c478bd9Sstevel@tonic-gate  * IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
177c478bd9Sstevel@tonic-gate  * implied. See the License for the specific language governing
187c478bd9Sstevel@tonic-gate  * rights and limitations under the License.
197c478bd9Sstevel@tonic-gate  *
207c478bd9Sstevel@tonic-gate  * The Original Code is Mozilla Communicator client code, released
217c478bd9Sstevel@tonic-gate  * March 31, 1998.
227c478bd9Sstevel@tonic-gate  *
237c478bd9Sstevel@tonic-gate  * The Initial Developer of the Original Code is Netscape
247c478bd9Sstevel@tonic-gate  * Communications Corporation. Portions created by Netscape are
257c478bd9Sstevel@tonic-gate  * Copyright (C) 1998-1999 Netscape Communications Corporation. All
267c478bd9Sstevel@tonic-gate  * Rights Reserved.
277c478bd9Sstevel@tonic-gate  *
287c478bd9Sstevel@tonic-gate  * Contributor(s):
297c478bd9Sstevel@tonic-gate  */
307c478bd9Sstevel@tonic-gate /*
317c478bd9Sstevel@tonic-gate  * Copyright (c) 1993, 1994 Regents of the University of Michigan.
327c478bd9Sstevel@tonic-gate  * All rights reserved.
337c478bd9Sstevel@tonic-gate  *
347c478bd9Sstevel@tonic-gate  * Redistribution and use in source and binary forms are permitted
357c478bd9Sstevel@tonic-gate  * provided that this notice is preserved and that due credit is given
367c478bd9Sstevel@tonic-gate  * to the University of Michigan at Ann Arbor. The name of the University
377c478bd9Sstevel@tonic-gate  * may not be used to endorse or promote products derived from this
387c478bd9Sstevel@tonic-gate  * software without specific prior written permission. This software
397c478bd9Sstevel@tonic-gate  * is provided ``as is'' without express or implied warranty.
407c478bd9Sstevel@tonic-gate  */
417c478bd9Sstevel@tonic-gate /*
427c478bd9Sstevel@tonic-gate  * disptmpl.c:  display template library routines for LDAP clients
437c478bd9Sstevel@tonic-gate  */
447c478bd9Sstevel@tonic-gate 
457c478bd9Sstevel@tonic-gate #include "ldap-int.h"
467c478bd9Sstevel@tonic-gate #include "disptmpl.h"
477c478bd9Sstevel@tonic-gate 
487c478bd9Sstevel@tonic-gate static void free_disptmpl( struct ldap_disptmpl *tmpl );
497c478bd9Sstevel@tonic-gate static int read_next_tmpl( char **bufp, long *blenp,
507c478bd9Sstevel@tonic-gate 	struct ldap_disptmpl **tmplp, int dtversion );
517c478bd9Sstevel@tonic-gate 
527c478bd9Sstevel@tonic-gate static char		*tmploptions[] = {
537c478bd9Sstevel@tonic-gate     "addable", "modrdn",
547c478bd9Sstevel@tonic-gate     "altview",
557c478bd9Sstevel@tonic-gate     NULL
567c478bd9Sstevel@tonic-gate };
577c478bd9Sstevel@tonic-gate 
587c478bd9Sstevel@tonic-gate 
597c478bd9Sstevel@tonic-gate static unsigned long	tmploptvals[] = {
607c478bd9Sstevel@tonic-gate     LDAP_DTMPL_OPT_ADDABLE, LDAP_DTMPL_OPT_ALLOWMODRDN,
617c478bd9Sstevel@tonic-gate     LDAP_DTMPL_OPT_ALTVIEW,
627c478bd9Sstevel@tonic-gate };
637c478bd9Sstevel@tonic-gate 
647c478bd9Sstevel@tonic-gate 
657c478bd9Sstevel@tonic-gate static char		*itemtypes[] = {
667c478bd9Sstevel@tonic-gate     "cis",			"mls",			"dn",
677c478bd9Sstevel@tonic-gate     "bool",			"jpeg",			"jpegbtn",
687c478bd9Sstevel@tonic-gate     "fax",			"faxbtn",		"audiobtn",
697c478bd9Sstevel@tonic-gate     "time",			"date",			"url",
707c478bd9Sstevel@tonic-gate     "searchact",		"linkact",		"adddnact",
717c478bd9Sstevel@tonic-gate     "addact",			"verifyact",		"mail",
727c478bd9Sstevel@tonic-gate     NULL
737c478bd9Sstevel@tonic-gate };
747c478bd9Sstevel@tonic-gate 
757c478bd9Sstevel@tonic-gate static unsigned long	itemsynids[] = {
767c478bd9Sstevel@tonic-gate     LDAP_SYN_CASEIGNORESTR,	LDAP_SYN_MULTILINESTR,	LDAP_SYN_DN,
777c478bd9Sstevel@tonic-gate     LDAP_SYN_BOOLEAN,		LDAP_SYN_JPEGIMAGE,	LDAP_SYN_JPEGBUTTON,
787c478bd9Sstevel@tonic-gate     LDAP_SYN_FAXIMAGE,		LDAP_SYN_FAXBUTTON,	LDAP_SYN_AUDIOBUTTON,
797c478bd9Sstevel@tonic-gate     LDAP_SYN_TIME,		LDAP_SYN_DATE,		LDAP_SYN_LABELEDURL,
807c478bd9Sstevel@tonic-gate     LDAP_SYN_SEARCHACTION,	LDAP_SYN_LINKACTION,	LDAP_SYN_ADDDNACTION,
817c478bd9Sstevel@tonic-gate     LDAP_SYN_ADDDNACTION,	LDAP_SYN_VERIFYDNACTION,LDAP_SYN_RFC822ADDR,
827c478bd9Sstevel@tonic-gate };
837c478bd9Sstevel@tonic-gate 
847c478bd9Sstevel@tonic-gate 
857c478bd9Sstevel@tonic-gate static char		*itemoptions[] = {
867c478bd9Sstevel@tonic-gate     "ro",		       		"sort",
877c478bd9Sstevel@tonic-gate     "1val",				"hide",
887c478bd9Sstevel@tonic-gate     "required",				"hideiffalse",
897c478bd9Sstevel@tonic-gate     NULL
907c478bd9Sstevel@tonic-gate };
917c478bd9Sstevel@tonic-gate 
927c478bd9Sstevel@tonic-gate 
937c478bd9Sstevel@tonic-gate static unsigned long	itemoptvals[] = {
947c478bd9Sstevel@tonic-gate     LDAP_DITEM_OPT_READONLY,		LDAP_DITEM_OPT_SORTVALUES,
957c478bd9Sstevel@tonic-gate     LDAP_DITEM_OPT_SINGLEVALUED,	LDAP_DITEM_OPT_HIDEIFEMPTY,
967c478bd9Sstevel@tonic-gate     LDAP_DITEM_OPT_VALUEREQUIRED,	LDAP_DITEM_OPT_HIDEIFFALSE,
977c478bd9Sstevel@tonic-gate };
987c478bd9Sstevel@tonic-gate 
997c478bd9Sstevel@tonic-gate 
1007c478bd9Sstevel@tonic-gate #define ADDEF_CONSTANT	"constant"
1017c478bd9Sstevel@tonic-gate #define ADDEF_ADDERSDN	"addersdn"
1027c478bd9Sstevel@tonic-gate 
1037c478bd9Sstevel@tonic-gate 
1047c478bd9Sstevel@tonic-gate int
1057c478bd9Sstevel@tonic-gate LDAP_CALL
ldap_init_templates(char * file,struct ldap_disptmpl ** tmpllistp)1067c478bd9Sstevel@tonic-gate ldap_init_templates( char *file, struct ldap_disptmpl **tmpllistp )
1077c478bd9Sstevel@tonic-gate {
1087c478bd9Sstevel@tonic-gate     FILE	*fp;
1097c478bd9Sstevel@tonic-gate     char	*buf;
1107c478bd9Sstevel@tonic-gate     long	rlen, len;
1117c478bd9Sstevel@tonic-gate     int		rc, eof;
1127c478bd9Sstevel@tonic-gate 
1137c478bd9Sstevel@tonic-gate     *tmpllistp = NULLDISPTMPL;
1147c478bd9Sstevel@tonic-gate 
115*004388ebScasper     if (( fp = fopen( file, "rF" )) == NULL ) {
1167c478bd9Sstevel@tonic-gate 	return( LDAP_TMPL_ERR_FILE );
1177c478bd9Sstevel@tonic-gate     }
1187c478bd9Sstevel@tonic-gate 
1197c478bd9Sstevel@tonic-gate     if ( fseek( fp, 0L, SEEK_END ) != 0 ) {	/* move to end to get len */
1207c478bd9Sstevel@tonic-gate 	fclose( fp );
1217c478bd9Sstevel@tonic-gate 	return( LDAP_TMPL_ERR_FILE );
1227c478bd9Sstevel@tonic-gate     }
1237c478bd9Sstevel@tonic-gate 
1247c478bd9Sstevel@tonic-gate     len = ftell( fp );
1257c478bd9Sstevel@tonic-gate 
1267c478bd9Sstevel@tonic-gate     if ( fseek( fp, 0L, SEEK_SET ) != 0 ) {	/* back to start of file */
1277c478bd9Sstevel@tonic-gate 	fclose( fp );
1287c478bd9Sstevel@tonic-gate 	return( LDAP_TMPL_ERR_FILE );
1297c478bd9Sstevel@tonic-gate     }
1307c478bd9Sstevel@tonic-gate 
1317c478bd9Sstevel@tonic-gate     if (( buf = NSLDAPI_MALLOC( (size_t)len )) == NULL ) {
1327c478bd9Sstevel@tonic-gate 	fclose( fp );
1337c478bd9Sstevel@tonic-gate 	return( LDAP_TMPL_ERR_MEM );
1347c478bd9Sstevel@tonic-gate     }
1357c478bd9Sstevel@tonic-gate 
1367c478bd9Sstevel@tonic-gate     rlen = fread( buf, 1, (size_t)len, fp );
1377c478bd9Sstevel@tonic-gate     eof = feof( fp );
1387c478bd9Sstevel@tonic-gate     fclose( fp );
1397c478bd9Sstevel@tonic-gate 
1407c478bd9Sstevel@tonic-gate     if ( rlen != len && !eof ) {	/* error:  didn't get the whole file */
1417c478bd9Sstevel@tonic-gate 	NSLDAPI_FREE( buf );
1427c478bd9Sstevel@tonic-gate 	return( LDAP_TMPL_ERR_FILE );
1437c478bd9Sstevel@tonic-gate     }
1447c478bd9Sstevel@tonic-gate 
1457c478bd9Sstevel@tonic-gate     rc = ldap_init_templates_buf( buf, rlen, tmpllistp );
1467c478bd9Sstevel@tonic-gate     NSLDAPI_FREE( buf );
1477c478bd9Sstevel@tonic-gate 
1487c478bd9Sstevel@tonic-gate     return( rc );
1497c478bd9Sstevel@tonic-gate }
1507c478bd9Sstevel@tonic-gate 
1517c478bd9Sstevel@tonic-gate 
1527c478bd9Sstevel@tonic-gate int
1537c478bd9Sstevel@tonic-gate LDAP_CALL
ldap_init_templates_buf(char * buf,long buflen,struct ldap_disptmpl ** tmpllistp)1547c478bd9Sstevel@tonic-gate ldap_init_templates_buf( char *buf, long buflen,
1557c478bd9Sstevel@tonic-gate 	struct ldap_disptmpl **tmpllistp )
1567c478bd9Sstevel@tonic-gate {
1577c478bd9Sstevel@tonic-gate     int				rc = 0, version;
1587c478bd9Sstevel@tonic-gate     char			**toks;
1597c478bd9Sstevel@tonic-gate     struct ldap_disptmpl	*prevtmpl, *tmpl;
1607c478bd9Sstevel@tonic-gate 
1617c478bd9Sstevel@tonic-gate     *tmpllistp = prevtmpl = NULLDISPTMPL;
1627c478bd9Sstevel@tonic-gate 
1637c478bd9Sstevel@tonic-gate     if ( ldap_next_line_tokens( &buf, &buflen, &toks ) != 2 ||
1647c478bd9Sstevel@tonic-gate 	    strcasecmp( toks[ 0 ], "version" ) != 0 ) {
1657c478bd9Sstevel@tonic-gate 	ldap_free_strarray( toks );
1667c478bd9Sstevel@tonic-gate 	return( LDAP_TMPL_ERR_SYNTAX );
1677c478bd9Sstevel@tonic-gate     }
1687c478bd9Sstevel@tonic-gate     version = atoi( toks[ 1 ] );
1697c478bd9Sstevel@tonic-gate     ldap_free_strarray( toks );
1707c478bd9Sstevel@tonic-gate     if ( version != LDAP_TEMPLATE_VERSION ) {
1717c478bd9Sstevel@tonic-gate 	return( LDAP_TMPL_ERR_VERSION );
1727c478bd9Sstevel@tonic-gate     }
1737c478bd9Sstevel@tonic-gate 
1747c478bd9Sstevel@tonic-gate     while ( buflen > 0 && ( rc = read_next_tmpl( &buf, &buflen, &tmpl,
1757c478bd9Sstevel@tonic-gate 	    version )) == 0 && tmpl != NULLDISPTMPL ) {
1767c478bd9Sstevel@tonic-gate 	if ( prevtmpl == NULLDISPTMPL ) {
1777c478bd9Sstevel@tonic-gate 	    *tmpllistp = tmpl;
1787c478bd9Sstevel@tonic-gate 	} else {
1797c478bd9Sstevel@tonic-gate 	    prevtmpl->dt_next = tmpl;
1807c478bd9Sstevel@tonic-gate 	}
1817c478bd9Sstevel@tonic-gate 	prevtmpl = tmpl;
1827c478bd9Sstevel@tonic-gate     }
1837c478bd9Sstevel@tonic-gate 
1847c478bd9Sstevel@tonic-gate     if ( rc != 0 ) {
1857c478bd9Sstevel@tonic-gate 	ldap_free_templates( *tmpllistp );
1867c478bd9Sstevel@tonic-gate     }
1877c478bd9Sstevel@tonic-gate 
1887c478bd9Sstevel@tonic-gate     return( rc );
1897c478bd9Sstevel@tonic-gate }
1907c478bd9Sstevel@tonic-gate 
1917c478bd9Sstevel@tonic-gate 
1927c478bd9Sstevel@tonic-gate 
1937c478bd9Sstevel@tonic-gate void
1947c478bd9Sstevel@tonic-gate LDAP_CALL
ldap_free_templates(struct ldap_disptmpl * tmpllist)1957c478bd9Sstevel@tonic-gate ldap_free_templates( struct ldap_disptmpl *tmpllist )
1967c478bd9Sstevel@tonic-gate {
1977c478bd9Sstevel@tonic-gate     struct ldap_disptmpl	*tp, *nexttp;
1987c478bd9Sstevel@tonic-gate 
1997c478bd9Sstevel@tonic-gate     if ( tmpllist != NULL ) {
2007c478bd9Sstevel@tonic-gate 	for ( tp = tmpllist; tp != NULL; tp = nexttp ) {
2017c478bd9Sstevel@tonic-gate 	    nexttp = tp->dt_next;
2027c478bd9Sstevel@tonic-gate 	    free_disptmpl( tp );
2037c478bd9Sstevel@tonic-gate 	}
2047c478bd9Sstevel@tonic-gate     }
2057c478bd9Sstevel@tonic-gate }
2067c478bd9Sstevel@tonic-gate 
2077c478bd9Sstevel@tonic-gate 
2087c478bd9Sstevel@tonic-gate static void
free_disptmpl(struct ldap_disptmpl * tmpl)2097c478bd9Sstevel@tonic-gate free_disptmpl( struct ldap_disptmpl *tmpl )
2107c478bd9Sstevel@tonic-gate {
2117c478bd9Sstevel@tonic-gate     if ( tmpl != NULL ) {
2127c478bd9Sstevel@tonic-gate 	if ( tmpl->dt_name != NULL ) {
2137c478bd9Sstevel@tonic-gate 	    NSLDAPI_FREE(  tmpl->dt_name );
2147c478bd9Sstevel@tonic-gate 	}
2157c478bd9Sstevel@tonic-gate 
2167c478bd9Sstevel@tonic-gate 	if ( tmpl->dt_pluralname != NULL ) {
2177c478bd9Sstevel@tonic-gate 	    NSLDAPI_FREE( tmpl->dt_pluralname );
2187c478bd9Sstevel@tonic-gate 	}
2197c478bd9Sstevel@tonic-gate 
2207c478bd9Sstevel@tonic-gate 	if ( tmpl->dt_iconname != NULL ) {
2217c478bd9Sstevel@tonic-gate 	    NSLDAPI_FREE( tmpl->dt_iconname );
2227c478bd9Sstevel@tonic-gate 	}
2237c478bd9Sstevel@tonic-gate 
2247c478bd9Sstevel@tonic-gate 	if ( tmpl->dt_authattrname != NULL ) {
2257c478bd9Sstevel@tonic-gate 	    NSLDAPI_FREE( tmpl->dt_authattrname );
2267c478bd9Sstevel@tonic-gate 	}
2277c478bd9Sstevel@tonic-gate 
2287c478bd9Sstevel@tonic-gate 	if ( tmpl->dt_defrdnattrname != NULL ) {
2297c478bd9Sstevel@tonic-gate 	    NSLDAPI_FREE( tmpl->dt_defrdnattrname );
2307c478bd9Sstevel@tonic-gate 	}
2317c478bd9Sstevel@tonic-gate 
2327c478bd9Sstevel@tonic-gate 	if ( tmpl->dt_defaddlocation != NULL ) {
2337c478bd9Sstevel@tonic-gate 	    NSLDAPI_FREE( tmpl->dt_defaddlocation );
2347c478bd9Sstevel@tonic-gate 	}
2357c478bd9Sstevel@tonic-gate 
2367c478bd9Sstevel@tonic-gate 	if (  tmpl->dt_oclist != NULL ) {
2377c478bd9Sstevel@tonic-gate 	    struct ldap_oclist	*ocp, *nextocp;
2387c478bd9Sstevel@tonic-gate 
2397c478bd9Sstevel@tonic-gate 	    for ( ocp = tmpl->dt_oclist; ocp != NULL; ocp = nextocp ) {
2407c478bd9Sstevel@tonic-gate 		nextocp = ocp->oc_next;
2417c478bd9Sstevel@tonic-gate 		ldap_free_strarray( ocp->oc_objclasses );
2427c478bd9Sstevel@tonic-gate 		NSLDAPI_FREE( ocp );
2437c478bd9Sstevel@tonic-gate 	    }
2447c478bd9Sstevel@tonic-gate 	}
2457c478bd9Sstevel@tonic-gate 
2467c478bd9Sstevel@tonic-gate 	if (  tmpl->dt_adddeflist != NULL ) {
2477c478bd9Sstevel@tonic-gate 	    struct ldap_adddeflist	*adp, *nextadp;
2487c478bd9Sstevel@tonic-gate 
2497c478bd9Sstevel@tonic-gate 	    for ( adp = tmpl->dt_adddeflist; adp != NULL; adp = nextadp ) {
2507c478bd9Sstevel@tonic-gate 		nextadp = adp->ad_next;
2517c478bd9Sstevel@tonic-gate 		if( adp->ad_attrname != NULL ) {
2527c478bd9Sstevel@tonic-gate 		    NSLDAPI_FREE( adp->ad_attrname );
2537c478bd9Sstevel@tonic-gate 		}
2547c478bd9Sstevel@tonic-gate 		if( adp->ad_value != NULL ) {
2557c478bd9Sstevel@tonic-gate 		    NSLDAPI_FREE( adp->ad_value );
2567c478bd9Sstevel@tonic-gate 		}
2577c478bd9Sstevel@tonic-gate 		NSLDAPI_FREE( adp );
2587c478bd9Sstevel@tonic-gate 	    }
2597c478bd9Sstevel@tonic-gate 	}
2607c478bd9Sstevel@tonic-gate 
2617c478bd9Sstevel@tonic-gate 	if (  tmpl->dt_items != NULL ) {
2627c478bd9Sstevel@tonic-gate 	    struct ldap_tmplitem	*rowp, *nextrowp, *colp, *nextcolp;
2637c478bd9Sstevel@tonic-gate 
2647c478bd9Sstevel@tonic-gate 	    for ( rowp = tmpl->dt_items; rowp != NULL; rowp = nextrowp ) {
2657c478bd9Sstevel@tonic-gate 		nextrowp = rowp->ti_next_in_col;
2667c478bd9Sstevel@tonic-gate 		for ( colp = rowp; colp != NULL; colp = nextcolp ) {
2677c478bd9Sstevel@tonic-gate 		    nextcolp = colp->ti_next_in_row;
2687c478bd9Sstevel@tonic-gate 		    if ( colp->ti_attrname != NULL ) {
2697c478bd9Sstevel@tonic-gate 			NSLDAPI_FREE( colp->ti_attrname );
2707c478bd9Sstevel@tonic-gate 		    }
2717c478bd9Sstevel@tonic-gate 		    if ( colp->ti_label != NULL ) {
2727c478bd9Sstevel@tonic-gate 			NSLDAPI_FREE( colp->ti_label );
2737c478bd9Sstevel@tonic-gate 		    }
2747c478bd9Sstevel@tonic-gate 		    if ( colp->ti_args != NULL ) {
2757c478bd9Sstevel@tonic-gate 			ldap_free_strarray( colp->ti_args );
2767c478bd9Sstevel@tonic-gate 		    }
2777c478bd9Sstevel@tonic-gate 		    NSLDAPI_FREE( colp );
2787c478bd9Sstevel@tonic-gate 		}
2797c478bd9Sstevel@tonic-gate 	    }
2807c478bd9Sstevel@tonic-gate 	}
2817c478bd9Sstevel@tonic-gate 
2827c478bd9Sstevel@tonic-gate 	NSLDAPI_FREE( tmpl );
2837c478bd9Sstevel@tonic-gate     }
2847c478bd9Sstevel@tonic-gate }
2857c478bd9Sstevel@tonic-gate 
2867c478bd9Sstevel@tonic-gate 
2877c478bd9Sstevel@tonic-gate struct ldap_disptmpl *
2887c478bd9Sstevel@tonic-gate LDAP_CALL
ldap_first_disptmpl(struct ldap_disptmpl * tmpllist)2897c478bd9Sstevel@tonic-gate ldap_first_disptmpl( struct ldap_disptmpl *tmpllist )
2907c478bd9Sstevel@tonic-gate {
2917c478bd9Sstevel@tonic-gate     return( tmpllist );
2927c478bd9Sstevel@tonic-gate }
2937c478bd9Sstevel@tonic-gate 
2947c478bd9Sstevel@tonic-gate 
2957c478bd9Sstevel@tonic-gate struct ldap_disptmpl *
2967c478bd9Sstevel@tonic-gate LDAP_CALL
ldap_next_disptmpl(struct ldap_disptmpl * tmpllist,struct ldap_disptmpl * tmpl)2977c478bd9Sstevel@tonic-gate ldap_next_disptmpl( struct ldap_disptmpl *tmpllist,
2987c478bd9Sstevel@tonic-gate 	struct ldap_disptmpl *tmpl )
2997c478bd9Sstevel@tonic-gate {
3007c478bd9Sstevel@tonic-gate     return( tmpl == NULLDISPTMPL ? tmpl : tmpl->dt_next );
3017c478bd9Sstevel@tonic-gate }
3027c478bd9Sstevel@tonic-gate 
3037c478bd9Sstevel@tonic-gate 
3047c478bd9Sstevel@tonic-gate struct ldap_disptmpl *
3057c478bd9Sstevel@tonic-gate LDAP_CALL
ldap_name2template(char * name,struct ldap_disptmpl * tmpllist)3067c478bd9Sstevel@tonic-gate ldap_name2template( char *name, struct ldap_disptmpl *tmpllist )
3077c478bd9Sstevel@tonic-gate {
3087c478bd9Sstevel@tonic-gate     struct ldap_disptmpl	*dtp;
3097c478bd9Sstevel@tonic-gate 
3107c478bd9Sstevel@tonic-gate     for ( dtp = ldap_first_disptmpl( tmpllist ); dtp != NULLDISPTMPL;
3117c478bd9Sstevel@tonic-gate 	    dtp = ldap_next_disptmpl( tmpllist, dtp )) {
3127c478bd9Sstevel@tonic-gate 	if ( strcasecmp( name, dtp->dt_name ) == 0 ) {
3137c478bd9Sstevel@tonic-gate 	    return( dtp );
3147c478bd9Sstevel@tonic-gate 	}
3157c478bd9Sstevel@tonic-gate     }
3167c478bd9Sstevel@tonic-gate 
3177c478bd9Sstevel@tonic-gate     return( NULLDISPTMPL );
3187c478bd9Sstevel@tonic-gate }
3197c478bd9Sstevel@tonic-gate 
3207c478bd9Sstevel@tonic-gate 
3217c478bd9Sstevel@tonic-gate struct ldap_disptmpl *
3227c478bd9Sstevel@tonic-gate LDAP_CALL
ldap_oc2template(char ** oclist,struct ldap_disptmpl * tmpllist)3237c478bd9Sstevel@tonic-gate ldap_oc2template( char **oclist, struct ldap_disptmpl *tmpllist )
3247c478bd9Sstevel@tonic-gate {
3257c478bd9Sstevel@tonic-gate     struct ldap_disptmpl	*dtp;
3267c478bd9Sstevel@tonic-gate     struct ldap_oclist		*oclp;
3277c478bd9Sstevel@tonic-gate     int				i, j, needcnt, matchcnt;
3287c478bd9Sstevel@tonic-gate 
3297c478bd9Sstevel@tonic-gate     if ( tmpllist == NULL || oclist == NULL || oclist[ 0 ] == NULL ) {
3307c478bd9Sstevel@tonic-gate 	return( NULLDISPTMPL );
3317c478bd9Sstevel@tonic-gate     }
3327c478bd9Sstevel@tonic-gate 
3337c478bd9Sstevel@tonic-gate     for ( dtp = ldap_first_disptmpl( tmpllist ); dtp != NULLDISPTMPL;
3347c478bd9Sstevel@tonic-gate 		dtp = ldap_next_disptmpl( tmpllist, dtp )) {
3357c478bd9Sstevel@tonic-gate 	for ( oclp = dtp->dt_oclist; oclp != NULLOCLIST;
3367c478bd9Sstevel@tonic-gate 		oclp = oclp->oc_next ) {
3377c478bd9Sstevel@tonic-gate 	    needcnt = matchcnt = 0;
3387c478bd9Sstevel@tonic-gate 	    for ( i = 0; oclp->oc_objclasses[ i ] != NULL; ++i ) {
3397c478bd9Sstevel@tonic-gate 		for ( j = 0; oclist[ j ] != NULL; ++j ) {
3407c478bd9Sstevel@tonic-gate 		    if ( strcasecmp( oclist[ j ], oclp->oc_objclasses[ i ] )
3417c478bd9Sstevel@tonic-gate 			    == 0 ) {
3427c478bd9Sstevel@tonic-gate 			++matchcnt;
3437c478bd9Sstevel@tonic-gate 		    }
3447c478bd9Sstevel@tonic-gate 		}
3457c478bd9Sstevel@tonic-gate 		++needcnt;
3467c478bd9Sstevel@tonic-gate 	    }
3477c478bd9Sstevel@tonic-gate 
3487c478bd9Sstevel@tonic-gate 	    if ( matchcnt == needcnt ) {
3497c478bd9Sstevel@tonic-gate 		return( dtp );
3507c478bd9Sstevel@tonic-gate 	    }
3517c478bd9Sstevel@tonic-gate 	}
3527c478bd9Sstevel@tonic-gate     }
3537c478bd9Sstevel@tonic-gate 
3547c478bd9Sstevel@tonic-gate     return( NULLDISPTMPL );
3557c478bd9Sstevel@tonic-gate }
3567c478bd9Sstevel@tonic-gate 
3577c478bd9Sstevel@tonic-gate 
3587c478bd9Sstevel@tonic-gate struct ldap_tmplitem *
3597c478bd9Sstevel@tonic-gate LDAP_CALL
ldap_first_tmplrow(struct ldap_disptmpl * tmpl)3607c478bd9Sstevel@tonic-gate ldap_first_tmplrow( struct ldap_disptmpl *tmpl )
3617c478bd9Sstevel@tonic-gate {
3627c478bd9Sstevel@tonic-gate     return( tmpl->dt_items );
3637c478bd9Sstevel@tonic-gate }
3647c478bd9Sstevel@tonic-gate 
3657c478bd9Sstevel@tonic-gate 
3667c478bd9Sstevel@tonic-gate struct ldap_tmplitem *
3677c478bd9Sstevel@tonic-gate LDAP_CALL
ldap_next_tmplrow(struct ldap_disptmpl * tmpl,struct ldap_tmplitem * row)3687c478bd9Sstevel@tonic-gate ldap_next_tmplrow( struct ldap_disptmpl *tmpl, struct ldap_tmplitem *row )
3697c478bd9Sstevel@tonic-gate {
3707c478bd9Sstevel@tonic-gate     return( row == NULLTMPLITEM ? row : row->ti_next_in_col );
3717c478bd9Sstevel@tonic-gate }
3727c478bd9Sstevel@tonic-gate 
3737c478bd9Sstevel@tonic-gate 
3747c478bd9Sstevel@tonic-gate struct ldap_tmplitem *
3757c478bd9Sstevel@tonic-gate LDAP_CALL
ldap_first_tmplcol(struct ldap_disptmpl * tmpl,struct ldap_tmplitem * row)3767c478bd9Sstevel@tonic-gate ldap_first_tmplcol( struct ldap_disptmpl *tmpl, struct ldap_tmplitem *row )
3777c478bd9Sstevel@tonic-gate {
3787c478bd9Sstevel@tonic-gate     return( row );
3797c478bd9Sstevel@tonic-gate }
3807c478bd9Sstevel@tonic-gate 
3817c478bd9Sstevel@tonic-gate 
3827c478bd9Sstevel@tonic-gate struct ldap_tmplitem *
3837c478bd9Sstevel@tonic-gate LDAP_CALL
ldap_next_tmplcol(struct ldap_disptmpl * tmpl,struct ldap_tmplitem * row,struct ldap_tmplitem * col)3847c478bd9Sstevel@tonic-gate ldap_next_tmplcol( struct ldap_disptmpl *tmpl, struct ldap_tmplitem *row,
3857c478bd9Sstevel@tonic-gate 	struct ldap_tmplitem *col )
3867c478bd9Sstevel@tonic-gate {
3877c478bd9Sstevel@tonic-gate     return( col == NULLTMPLITEM ? col : col->ti_next_in_row );
3887c478bd9Sstevel@tonic-gate }
3897c478bd9Sstevel@tonic-gate 
3907c478bd9Sstevel@tonic-gate 
3917c478bd9Sstevel@tonic-gate char **
3927c478bd9Sstevel@tonic-gate LDAP_CALL
ldap_tmplattrs(struct ldap_disptmpl * tmpl,char ** includeattrs,int exclude,unsigned long syntaxmask)3937c478bd9Sstevel@tonic-gate ldap_tmplattrs( struct ldap_disptmpl *tmpl, char **includeattrs,
3947c478bd9Sstevel@tonic-gate 	int exclude, unsigned long syntaxmask )
3957c478bd9Sstevel@tonic-gate {
3967c478bd9Sstevel@tonic-gate /*
3977c478bd9Sstevel@tonic-gate  * this routine should filter out duplicate attributes...
3987c478bd9Sstevel@tonic-gate  */
3997c478bd9Sstevel@tonic-gate     struct ldap_tmplitem	*tirowp, *ticolp;
4007c478bd9Sstevel@tonic-gate     int			i, attrcnt, memerr;
4017c478bd9Sstevel@tonic-gate     char		**attrs;
4027c478bd9Sstevel@tonic-gate 
4037c478bd9Sstevel@tonic-gate     attrcnt = 0;
4047c478bd9Sstevel@tonic-gate     memerr = 0;
4057c478bd9Sstevel@tonic-gate 
4067c478bd9Sstevel@tonic-gate     if (( attrs = (char **)NSLDAPI_MALLOC( sizeof( char * ))) == NULL ) {
4077c478bd9Sstevel@tonic-gate 	return( NULL );
4087c478bd9Sstevel@tonic-gate     }
4097c478bd9Sstevel@tonic-gate 
4107c478bd9Sstevel@tonic-gate     if ( includeattrs != NULL ) {
4117c478bd9Sstevel@tonic-gate 	for ( i = 0; !memerr && includeattrs[ i ] != NULL; ++i ) {
4127c478bd9Sstevel@tonic-gate 	    if (( attrs = (char **)NSLDAPI_REALLOC( attrs, ( attrcnt + 2 ) *
4137c478bd9Sstevel@tonic-gate 		    sizeof( char * ))) == NULL || ( attrs[ attrcnt++ ] =
4147c478bd9Sstevel@tonic-gate 		    nsldapi_strdup( includeattrs[ i ] )) == NULL ) {
4157c478bd9Sstevel@tonic-gate 		memerr = 1;
4167c478bd9Sstevel@tonic-gate 	    } else {
4177c478bd9Sstevel@tonic-gate 		attrs[ attrcnt ] = NULL;
4187c478bd9Sstevel@tonic-gate 	    }
4197c478bd9Sstevel@tonic-gate 	}
4207c478bd9Sstevel@tonic-gate     }
4217c478bd9Sstevel@tonic-gate 
4227c478bd9Sstevel@tonic-gate     for ( tirowp = ldap_first_tmplrow( tmpl );
4237c478bd9Sstevel@tonic-gate 	    !memerr && tirowp != NULLTMPLITEM;
4247c478bd9Sstevel@tonic-gate 	    tirowp = ldap_next_tmplrow( tmpl, tirowp )) {
4257c478bd9Sstevel@tonic-gate 	for ( ticolp = ldap_first_tmplcol( tmpl, tirowp );
4267c478bd9Sstevel@tonic-gate 		ticolp != NULLTMPLITEM;
4277c478bd9Sstevel@tonic-gate 		ticolp = ldap_next_tmplcol( tmpl, tirowp, ticolp )) {
4287c478bd9Sstevel@tonic-gate 
4297c478bd9Sstevel@tonic-gate 	    if ( syntaxmask != 0 ) {
4307c478bd9Sstevel@tonic-gate 		if (( exclude &&
4317c478bd9Sstevel@tonic-gate 			( syntaxmask & ticolp->ti_syntaxid ) != 0 ) ||
4327c478bd9Sstevel@tonic-gate 			( !exclude &&
4337c478bd9Sstevel@tonic-gate 			( syntaxmask & ticolp->ti_syntaxid ) == 0 )) {
4347c478bd9Sstevel@tonic-gate 		    continue;
4357c478bd9Sstevel@tonic-gate 		}
4367c478bd9Sstevel@tonic-gate 	    }
4377c478bd9Sstevel@tonic-gate 
4387c478bd9Sstevel@tonic-gate 	    if ( ticolp->ti_attrname != NULL ) {
4397c478bd9Sstevel@tonic-gate 		if (( attrs = (char **)NSLDAPI_REALLOC( attrs, ( attrcnt + 2 )
4407c478bd9Sstevel@tonic-gate 			* sizeof( char * ))) == NULL || ( attrs[ attrcnt++ ] =
4417c478bd9Sstevel@tonic-gate 			nsldapi_strdup( ticolp->ti_attrname )) == NULL ) {
4427c478bd9Sstevel@tonic-gate 		    memerr = 1;
4437c478bd9Sstevel@tonic-gate 		} else {
4447c478bd9Sstevel@tonic-gate 		    attrs[ attrcnt ] = NULL;
4457c478bd9Sstevel@tonic-gate 		}
4467c478bd9Sstevel@tonic-gate 	    }
4477c478bd9Sstevel@tonic-gate 	}
4487c478bd9Sstevel@tonic-gate     }
4497c478bd9Sstevel@tonic-gate 
4507c478bd9Sstevel@tonic-gate     if ( memerr || attrcnt == 0 ) {
4517c478bd9Sstevel@tonic-gate 	for ( i = 0; i < attrcnt; ++i ) {
4527c478bd9Sstevel@tonic-gate 	    if ( attrs[ i ] != NULL ) {
4537c478bd9Sstevel@tonic-gate 		NSLDAPI_FREE( attrs[ i ] );
4547c478bd9Sstevel@tonic-gate 	    }
4557c478bd9Sstevel@tonic-gate 	}
4567c478bd9Sstevel@tonic-gate 
4577c478bd9Sstevel@tonic-gate 	NSLDAPI_FREE( (char *)attrs );
4587c478bd9Sstevel@tonic-gate 	return( NULL );
4597c478bd9Sstevel@tonic-gate     }
4607c478bd9Sstevel@tonic-gate 
4617c478bd9Sstevel@tonic-gate     return( attrs );
4627c478bd9Sstevel@tonic-gate }
4637c478bd9Sstevel@tonic-gate 
4647c478bd9Sstevel@tonic-gate 
4657c478bd9Sstevel@tonic-gate static int
read_next_tmpl(char ** bufp,long * blenp,struct ldap_disptmpl ** tmplp,int dtversion)4667c478bd9Sstevel@tonic-gate read_next_tmpl( char **bufp, long *blenp, struct ldap_disptmpl **tmplp,
4677c478bd9Sstevel@tonic-gate 	int dtversion )
4687c478bd9Sstevel@tonic-gate {
4697c478bd9Sstevel@tonic-gate     int				i, j, tokcnt, samerow, adsource;
4707c478bd9Sstevel@tonic-gate     char			**toks, *itemopts;
4717c478bd9Sstevel@tonic-gate     struct ldap_disptmpl	*tmpl = NULL;
4727c478bd9Sstevel@tonic-gate     struct ldap_oclist		*ocp = NULL, *prevocp = NULL;
4737c478bd9Sstevel@tonic-gate     struct ldap_adddeflist	*adp = NULL, *prevadp = NULL;
4747c478bd9Sstevel@tonic-gate     struct ldap_tmplitem	*rowp = NULL, *ip = NULL, *previp = NULL;
4757c478bd9Sstevel@tonic-gate 
4767c478bd9Sstevel@tonic-gate     /*
4777c478bd9Sstevel@tonic-gate      * template name comes first
4787c478bd9Sstevel@tonic-gate      */
4797c478bd9Sstevel@tonic-gate     if (( tokcnt = ldap_next_line_tokens( bufp, blenp, &toks )) != 1 ) {
4807c478bd9Sstevel@tonic-gate 	ldap_free_strarray( toks );
4817c478bd9Sstevel@tonic-gate 	return( tokcnt == 0 ? 0 : LDAP_TMPL_ERR_SYNTAX );
4827c478bd9Sstevel@tonic-gate     }
4837c478bd9Sstevel@tonic-gate 
4847c478bd9Sstevel@tonic-gate     if (( tmpl = (struct ldap_disptmpl *)NSLDAPI_CALLOC( 1,
4857c478bd9Sstevel@tonic-gate 	    sizeof( struct ldap_disptmpl ))) == NULL ) {
4867c478bd9Sstevel@tonic-gate 	ldap_free_strarray( toks );
4877c478bd9Sstevel@tonic-gate 	return(  LDAP_TMPL_ERR_MEM );
4887c478bd9Sstevel@tonic-gate     }
4897c478bd9Sstevel@tonic-gate     tmpl->dt_name = toks[ 0 ];
4907c478bd9Sstevel@tonic-gate     NSLDAPI_FREE( (char *)toks );
4917c478bd9Sstevel@tonic-gate 
4927c478bd9Sstevel@tonic-gate     /*
4937c478bd9Sstevel@tonic-gate      * template plural name comes next
4947c478bd9Sstevel@tonic-gate      */
4957c478bd9Sstevel@tonic-gate     if (( tokcnt = ldap_next_line_tokens( bufp, blenp, &toks )) != 1 ) {
4967c478bd9Sstevel@tonic-gate 	ldap_free_strarray( toks );
4977c478bd9Sstevel@tonic-gate 	free_disptmpl( tmpl );
4987c478bd9Sstevel@tonic-gate 	return( LDAP_TMPL_ERR_SYNTAX );
4997c478bd9Sstevel@tonic-gate     }
5007c478bd9Sstevel@tonic-gate     tmpl->dt_pluralname = toks[ 0 ];
5017c478bd9Sstevel@tonic-gate     NSLDAPI_FREE( (char *)toks );
5027c478bd9Sstevel@tonic-gate 
5037c478bd9Sstevel@tonic-gate     /*
5047c478bd9Sstevel@tonic-gate      * template icon name is next
5057c478bd9Sstevel@tonic-gate      */
5067c478bd9Sstevel@tonic-gate     if (( tokcnt = ldap_next_line_tokens( bufp, blenp, &toks )) != 1 ) {
5077c478bd9Sstevel@tonic-gate 	ldap_free_strarray( toks );
5087c478bd9Sstevel@tonic-gate 	free_disptmpl( tmpl );
5097c478bd9Sstevel@tonic-gate 	return( LDAP_TMPL_ERR_SYNTAX );
5107c478bd9Sstevel@tonic-gate     }
5117c478bd9Sstevel@tonic-gate     tmpl->dt_iconname = toks[ 0 ];
5127c478bd9Sstevel@tonic-gate     NSLDAPI_FREE( (char *)toks );
5137c478bd9Sstevel@tonic-gate 
5147c478bd9Sstevel@tonic-gate     /*
5157c478bd9Sstevel@tonic-gate      * template options come next
5167c478bd9Sstevel@tonic-gate      */
5177c478bd9Sstevel@tonic-gate     if (( tokcnt = ldap_next_line_tokens( bufp, blenp, &toks )) < 1 ) {
5187c478bd9Sstevel@tonic-gate 	ldap_free_strarray( toks );
5197c478bd9Sstevel@tonic-gate 	free_disptmpl( tmpl );
5207c478bd9Sstevel@tonic-gate 	return( LDAP_TMPL_ERR_SYNTAX );
5217c478bd9Sstevel@tonic-gate     }
5227c478bd9Sstevel@tonic-gate     for ( i = 0; toks[ i ] != NULL; ++i ) {
5237c478bd9Sstevel@tonic-gate 	for ( j = 0; tmploptions[ j ] != NULL; ++j ) {
5247c478bd9Sstevel@tonic-gate 	    if ( strcasecmp( toks[ i ], tmploptions[ j ] ) == 0 ) {
5257c478bd9Sstevel@tonic-gate 		tmpl->dt_options |= tmploptvals[ j ];
5267c478bd9Sstevel@tonic-gate 	    }
5277c478bd9Sstevel@tonic-gate 	}
5287c478bd9Sstevel@tonic-gate     }
5297c478bd9Sstevel@tonic-gate     ldap_free_strarray( toks );
5307c478bd9Sstevel@tonic-gate 
5317c478bd9Sstevel@tonic-gate     /*
5327c478bd9Sstevel@tonic-gate      * object class list is next
5337c478bd9Sstevel@tonic-gate      */
5347c478bd9Sstevel@tonic-gate     while (( tokcnt = ldap_next_line_tokens( bufp, blenp, &toks )) > 0 ) {
5357c478bd9Sstevel@tonic-gate 	if (( ocp = (struct ldap_oclist *)NSLDAPI_CALLOC( 1,
5367c478bd9Sstevel@tonic-gate 		sizeof( struct ldap_oclist ))) == NULL ) {
5377c478bd9Sstevel@tonic-gate 	    ldap_free_strarray( toks );
5387c478bd9Sstevel@tonic-gate 	    free_disptmpl( tmpl );
5397c478bd9Sstevel@tonic-gate 	    return( LDAP_TMPL_ERR_MEM );
5407c478bd9Sstevel@tonic-gate 	}
5417c478bd9Sstevel@tonic-gate 	ocp->oc_objclasses = toks;
5427c478bd9Sstevel@tonic-gate 	if ( tmpl->dt_oclist == NULL ) {
5437c478bd9Sstevel@tonic-gate 	    tmpl->dt_oclist = ocp;
5447c478bd9Sstevel@tonic-gate 	} else {
5457c478bd9Sstevel@tonic-gate 	    prevocp->oc_next = ocp;
5467c478bd9Sstevel@tonic-gate 	}
5477c478bd9Sstevel@tonic-gate 	prevocp = ocp;
5487c478bd9Sstevel@tonic-gate     }
5497c478bd9Sstevel@tonic-gate     if ( tokcnt < 0 ) {
5507c478bd9Sstevel@tonic-gate 	free_disptmpl( tmpl );
5517c478bd9Sstevel@tonic-gate 	return( LDAP_TMPL_ERR_SYNTAX );
5527c478bd9Sstevel@tonic-gate     }
5537c478bd9Sstevel@tonic-gate 
5547c478bd9Sstevel@tonic-gate     /*
5557c478bd9Sstevel@tonic-gate      * read name of attribute to authenticate as
5567c478bd9Sstevel@tonic-gate      */
5577c478bd9Sstevel@tonic-gate     if (( tokcnt = ldap_next_line_tokens( bufp, blenp, &toks )) != 1 ) {
5587c478bd9Sstevel@tonic-gate 	ldap_free_strarray( toks );
5597c478bd9Sstevel@tonic-gate 	free_disptmpl( tmpl );
5607c478bd9Sstevel@tonic-gate 	return( LDAP_TMPL_ERR_SYNTAX );
5617c478bd9Sstevel@tonic-gate     }
5627c478bd9Sstevel@tonic-gate     if ( toks[ 0 ][ 0 ] != '\0' ) {
5637c478bd9Sstevel@tonic-gate 	tmpl->dt_authattrname = toks[ 0 ];
5647c478bd9Sstevel@tonic-gate     } else {
5657c478bd9Sstevel@tonic-gate 	NSLDAPI_FREE( toks[ 0 ] );
5667c478bd9Sstevel@tonic-gate     }
5677c478bd9Sstevel@tonic-gate     NSLDAPI_FREE( (char *)toks );
5687c478bd9Sstevel@tonic-gate 
5697c478bd9Sstevel@tonic-gate     /*
5707c478bd9Sstevel@tonic-gate      * read default attribute to use for RDN
5717c478bd9Sstevel@tonic-gate      */
5727c478bd9Sstevel@tonic-gate     if (( tokcnt = ldap_next_line_tokens( bufp, blenp, &toks )) != 1 ) {
5737c478bd9Sstevel@tonic-gate 	ldap_free_strarray( toks );
5747c478bd9Sstevel@tonic-gate 	free_disptmpl( tmpl );
5757c478bd9Sstevel@tonic-gate 	return( LDAP_TMPL_ERR_SYNTAX );
5767c478bd9Sstevel@tonic-gate     }
5777c478bd9Sstevel@tonic-gate     tmpl->dt_defrdnattrname = toks[ 0 ];
5787c478bd9Sstevel@tonic-gate     NSLDAPI_FREE( (char *)toks );
5797c478bd9Sstevel@tonic-gate 
5807c478bd9Sstevel@tonic-gate     /*
5817c478bd9Sstevel@tonic-gate      * read default location for new entries
5827c478bd9Sstevel@tonic-gate      */
5837c478bd9Sstevel@tonic-gate     if (( tokcnt = ldap_next_line_tokens( bufp, blenp, &toks )) != 1 ) {
5847c478bd9Sstevel@tonic-gate 	ldap_free_strarray( toks );
5857c478bd9Sstevel@tonic-gate 	free_disptmpl( tmpl );
5867c478bd9Sstevel@tonic-gate 	return( LDAP_TMPL_ERR_SYNTAX );
5877c478bd9Sstevel@tonic-gate     }
5887c478bd9Sstevel@tonic-gate     if ( toks[ 0 ][ 0 ] != '\0' ) {
5897c478bd9Sstevel@tonic-gate 	tmpl->dt_defaddlocation = toks[ 0 ];
5907c478bd9Sstevel@tonic-gate     } else {
5917c478bd9Sstevel@tonic-gate 	NSLDAPI_FREE( toks[ 0 ] );
5927c478bd9Sstevel@tonic-gate     }
5937c478bd9Sstevel@tonic-gate     NSLDAPI_FREE( (char *)toks );
5947c478bd9Sstevel@tonic-gate 
5957c478bd9Sstevel@tonic-gate     /*
5967c478bd9Sstevel@tonic-gate      * read list of rules used to define default values for new entries
5977c478bd9Sstevel@tonic-gate      */
5987c478bd9Sstevel@tonic-gate     while (( tokcnt = ldap_next_line_tokens( bufp, blenp, &toks )) > 0 ) {
5997c478bd9Sstevel@tonic-gate 	if ( strcasecmp( ADDEF_CONSTANT, toks[ 0 ] ) == 0 ) {
6007c478bd9Sstevel@tonic-gate 	    adsource = LDAP_ADSRC_CONSTANTVALUE;
6017c478bd9Sstevel@tonic-gate 	} else if ( strcasecmp( ADDEF_ADDERSDN, toks[ 0 ] ) == 0 ) {
6027c478bd9Sstevel@tonic-gate 	    adsource = LDAP_ADSRC_ADDERSDN;
6037c478bd9Sstevel@tonic-gate 	} else {
6047c478bd9Sstevel@tonic-gate 	    adsource = 0;
6057c478bd9Sstevel@tonic-gate 	}
6067c478bd9Sstevel@tonic-gate 	if ( adsource == 0 || tokcnt < 2 ||
6077c478bd9Sstevel@tonic-gate 		( adsource == LDAP_ADSRC_CONSTANTVALUE && tokcnt != 3 ) ||
6087c478bd9Sstevel@tonic-gate 		( adsource == LDAP_ADSRC_ADDERSDN && tokcnt != 2 )) {
6097c478bd9Sstevel@tonic-gate 	    ldap_free_strarray( toks );
6107c478bd9Sstevel@tonic-gate 	    free_disptmpl( tmpl );
6117c478bd9Sstevel@tonic-gate 	    return( LDAP_TMPL_ERR_SYNTAX );
6127c478bd9Sstevel@tonic-gate 	}
6137c478bd9Sstevel@tonic-gate 
6147c478bd9Sstevel@tonic-gate 	if (( adp = (struct ldap_adddeflist *)NSLDAPI_CALLOC( 1,
6157c478bd9Sstevel@tonic-gate 		sizeof( struct ldap_adddeflist ))) == NULL ) {
6167c478bd9Sstevel@tonic-gate 	    ldap_free_strarray( toks );
6177c478bd9Sstevel@tonic-gate 	    free_disptmpl( tmpl );
6187c478bd9Sstevel@tonic-gate 	    return( LDAP_TMPL_ERR_MEM );
6197c478bd9Sstevel@tonic-gate 	}
6207c478bd9Sstevel@tonic-gate 	adp->ad_source = adsource;
6217c478bd9Sstevel@tonic-gate 	adp->ad_attrname = toks[ 1 ];
6227c478bd9Sstevel@tonic-gate 	if ( adsource == LDAP_ADSRC_CONSTANTVALUE ) {
6237c478bd9Sstevel@tonic-gate 	    adp->ad_value = toks[ 2 ];
6247c478bd9Sstevel@tonic-gate 	}
6257c478bd9Sstevel@tonic-gate 	NSLDAPI_FREE( toks[ 0 ] );
6267c478bd9Sstevel@tonic-gate 	NSLDAPI_FREE( (char *)toks );
6277c478bd9Sstevel@tonic-gate 
6287c478bd9Sstevel@tonic-gate 	if ( tmpl->dt_adddeflist == NULL ) {
6297c478bd9Sstevel@tonic-gate 	    tmpl->dt_adddeflist = adp;
6307c478bd9Sstevel@tonic-gate 	} else {
6317c478bd9Sstevel@tonic-gate 	    prevadp->ad_next = adp;
6327c478bd9Sstevel@tonic-gate 	}
6337c478bd9Sstevel@tonic-gate 	prevadp = adp;
6347c478bd9Sstevel@tonic-gate     }
6357c478bd9Sstevel@tonic-gate 
6367c478bd9Sstevel@tonic-gate     /*
6377c478bd9Sstevel@tonic-gate      * item list is next
6387c478bd9Sstevel@tonic-gate      */
6397c478bd9Sstevel@tonic-gate     samerow = 0;
6407c478bd9Sstevel@tonic-gate     while (( tokcnt = ldap_next_line_tokens( bufp, blenp, &toks )) > 0 ) {
6417c478bd9Sstevel@tonic-gate 	if ( strcasecmp( toks[ 0 ], "item" ) == 0 ) {
6427c478bd9Sstevel@tonic-gate 	    if ( tokcnt < 4 ) {
6437c478bd9Sstevel@tonic-gate 		ldap_free_strarray( toks );
6447c478bd9Sstevel@tonic-gate 		free_disptmpl( tmpl );
6457c478bd9Sstevel@tonic-gate 		return( LDAP_TMPL_ERR_SYNTAX );
6467c478bd9Sstevel@tonic-gate 	    }
6477c478bd9Sstevel@tonic-gate 
6487c478bd9Sstevel@tonic-gate 	    if (( ip = (struct ldap_tmplitem *)NSLDAPI_CALLOC( 1,
6497c478bd9Sstevel@tonic-gate 		    sizeof( struct ldap_tmplitem ))) == NULL ) {
6507c478bd9Sstevel@tonic-gate 		ldap_free_strarray( toks );
6517c478bd9Sstevel@tonic-gate 		free_disptmpl( tmpl );
6527c478bd9Sstevel@tonic-gate 		return( LDAP_TMPL_ERR_MEM );
6537c478bd9Sstevel@tonic-gate 	    }
6547c478bd9Sstevel@tonic-gate 
6557c478bd9Sstevel@tonic-gate 	    /*
6567c478bd9Sstevel@tonic-gate 	     * find syntaxid from config file string
6577c478bd9Sstevel@tonic-gate 	     */
6587c478bd9Sstevel@tonic-gate 	    while (( itemopts = strrchr( toks[ 1 ], ',' )) != NULL ) {
6597c478bd9Sstevel@tonic-gate 		*itemopts++ = '\0';
6607c478bd9Sstevel@tonic-gate 		for ( i = 0; itemoptions[ i ] != NULL; ++i ) {
6617c478bd9Sstevel@tonic-gate 		    if ( strcasecmp( itemopts, itemoptions[ i ] ) == 0 ) {
6627c478bd9Sstevel@tonic-gate 			break;
6637c478bd9Sstevel@tonic-gate 		    }
6647c478bd9Sstevel@tonic-gate 		}
6657c478bd9Sstevel@tonic-gate 		if ( itemoptions[ i ] == NULL ) {
6667c478bd9Sstevel@tonic-gate 		    ldap_free_strarray( toks );
6677c478bd9Sstevel@tonic-gate 		    free_disptmpl( tmpl );
6687c478bd9Sstevel@tonic-gate 		    return( LDAP_TMPL_ERR_SYNTAX );
6697c478bd9Sstevel@tonic-gate 		}
6707c478bd9Sstevel@tonic-gate 		ip->ti_options |= itemoptvals[ i ];
6717c478bd9Sstevel@tonic-gate 	    }
6727c478bd9Sstevel@tonic-gate 
6737c478bd9Sstevel@tonic-gate 	    for ( i = 0; itemtypes[ i ] != NULL; ++i ) {
6747c478bd9Sstevel@tonic-gate 		if ( strcasecmp( toks[ 1 ], itemtypes[ i ] ) == 0 ) {
6757c478bd9Sstevel@tonic-gate 		    break;
6767c478bd9Sstevel@tonic-gate 		}
6777c478bd9Sstevel@tonic-gate 	    }
6787c478bd9Sstevel@tonic-gate 	    if ( itemtypes[ i ] == NULL ) {
6797c478bd9Sstevel@tonic-gate 		ldap_free_strarray( toks );
6807c478bd9Sstevel@tonic-gate 		free_disptmpl( tmpl );
6817c478bd9Sstevel@tonic-gate 		return( LDAP_TMPL_ERR_SYNTAX );
6827c478bd9Sstevel@tonic-gate 	    }
6837c478bd9Sstevel@tonic-gate 
6847c478bd9Sstevel@tonic-gate 	    NSLDAPI_FREE( toks[ 0 ] );
6857c478bd9Sstevel@tonic-gate 	    NSLDAPI_FREE( toks[ 1 ] );
6867c478bd9Sstevel@tonic-gate 	    ip->ti_syntaxid = itemsynids[ i ];
6877c478bd9Sstevel@tonic-gate 	    ip->ti_label = toks[ 2 ];
6887c478bd9Sstevel@tonic-gate 	    if ( toks[ 3 ][ 0 ] == '\0' ) {
6897c478bd9Sstevel@tonic-gate 		ip->ti_attrname = NULL;
6907c478bd9Sstevel@tonic-gate 		NSLDAPI_FREE( toks[ 3 ] );
6917c478bd9Sstevel@tonic-gate 	    } else {
6927c478bd9Sstevel@tonic-gate 		ip->ti_attrname = toks[ 3 ];
6937c478bd9Sstevel@tonic-gate 	    }
6947c478bd9Sstevel@tonic-gate 	    if ( toks[ 4 ] != NULL ) {	/* extra args. */
6957c478bd9Sstevel@tonic-gate 		for ( i = 0; toks[ i + 4 ] != NULL; ++i ) {
6967c478bd9Sstevel@tonic-gate 		    ;
6977c478bd9Sstevel@tonic-gate 		}
6987c478bd9Sstevel@tonic-gate 		if (( ip->ti_args = (char **)NSLDAPI_CALLOC( i + 1,
6997c478bd9Sstevel@tonic-gate 			sizeof( char * ))) == NULL ) {
7007c478bd9Sstevel@tonic-gate 		    free_disptmpl( tmpl );
7017c478bd9Sstevel@tonic-gate 		    return( LDAP_TMPL_ERR_MEM );
7027c478bd9Sstevel@tonic-gate 		}
7037c478bd9Sstevel@tonic-gate 		for ( i = 0; toks[ i + 4 ] != NULL; ++i ) {
7047c478bd9Sstevel@tonic-gate 		    ip->ti_args[ i ] = toks[ i + 4 ];
7057c478bd9Sstevel@tonic-gate 		}
7067c478bd9Sstevel@tonic-gate 	    }
7077c478bd9Sstevel@tonic-gate 	    NSLDAPI_FREE( (char *)toks );
7087c478bd9Sstevel@tonic-gate 
7097c478bd9Sstevel@tonic-gate 	    if ( tmpl->dt_items == NULL ) {
7107c478bd9Sstevel@tonic-gate 		tmpl->dt_items = rowp = ip;
7117c478bd9Sstevel@tonic-gate 	    } else if ( samerow ) {
7127c478bd9Sstevel@tonic-gate 		previp->ti_next_in_row = ip;
7137c478bd9Sstevel@tonic-gate 	    } else {
7147c478bd9Sstevel@tonic-gate 		rowp->ti_next_in_col = ip;
7157c478bd9Sstevel@tonic-gate 		rowp = ip;
7167c478bd9Sstevel@tonic-gate 	    }
7177c478bd9Sstevel@tonic-gate 	    previp = ip;
7187c478bd9Sstevel@tonic-gate 	    samerow = 0;
7197c478bd9Sstevel@tonic-gate 	} else if ( strcasecmp( toks[ 0 ], "samerow" ) == 0 ) {
7207c478bd9Sstevel@tonic-gate 	    ldap_free_strarray( toks );
7217c478bd9Sstevel@tonic-gate 	    samerow = 1;
7227c478bd9Sstevel@tonic-gate 	} else {
7237c478bd9Sstevel@tonic-gate 	    ldap_free_strarray( toks );
7247c478bd9Sstevel@tonic-gate 	    free_disptmpl( tmpl );
7257c478bd9Sstevel@tonic-gate 	    return( LDAP_TMPL_ERR_SYNTAX );
7267c478bd9Sstevel@tonic-gate 	}
7277c478bd9Sstevel@tonic-gate     }
7287c478bd9Sstevel@tonic-gate     if ( tokcnt < 0 ) {
7297c478bd9Sstevel@tonic-gate 	free_disptmpl( tmpl );
7307c478bd9Sstevel@tonic-gate 	return( LDAP_TMPL_ERR_SYNTAX );
7317c478bd9Sstevel@tonic-gate     }
7327c478bd9Sstevel@tonic-gate 
7337c478bd9Sstevel@tonic-gate     *tmplp = tmpl;
7347c478bd9Sstevel@tonic-gate     return( 0 );
7357c478bd9Sstevel@tonic-gate }
7367c478bd9Sstevel@tonic-gate 
7377c478bd9Sstevel@tonic-gate 
7387c478bd9Sstevel@tonic-gate struct tmplerror {
7397c478bd9Sstevel@tonic-gate 	int	e_code;
7407c478bd9Sstevel@tonic-gate 	char	*e_reason;
7417c478bd9Sstevel@tonic-gate };
7427c478bd9Sstevel@tonic-gate 
7437c478bd9Sstevel@tonic-gate #ifdef SUN
7447c478bd9Sstevel@tonic-gate static struct tmplerror ldap_tmplerrlist[] = {
7457c478bd9Sstevel@tonic-gate 	{ LDAP_TMPL_ERR_VERSION, 0},
7467c478bd9Sstevel@tonic-gate 	{ LDAP_TMPL_ERR_MEM,     0},
7477c478bd9Sstevel@tonic-gate 	{ LDAP_TMPL_ERR_SYNTAX,  0},
7487c478bd9Sstevel@tonic-gate 	{ LDAP_TMPL_ERR_FILE,    0},
7497c478bd9Sstevel@tonic-gate 	{ -1, 0 }
7507c478bd9Sstevel@tonic-gate };
7517c478bd9Sstevel@tonic-gate #else
7527c478bd9Sstevel@tonic-gate static struct tmplerror ldap_tmplerrlist[] = {
7537c478bd9Sstevel@tonic-gate 	{ LDAP_TMPL_ERR_VERSION, "Bad template version"		},
7547c478bd9Sstevel@tonic-gate 	{ LDAP_TMPL_ERR_MEM,     "Out of memory"		},
7557c478bd9Sstevel@tonic-gate 	{ LDAP_TMPL_ERR_SYNTAX,  "Bad template syntax"		},
7567c478bd9Sstevel@tonic-gate 	{ LDAP_TMPL_ERR_FILE,    "File error reading template"	},
7577c478bd9Sstevel@tonic-gate 	{ -1, 0 }
7587c478bd9Sstevel@tonic-gate };
7597c478bd9Sstevel@tonic-gate #endif
7607c478bd9Sstevel@tonic-gate 
7617c478bd9Sstevel@tonic-gate char *
7627c478bd9Sstevel@tonic-gate LDAP_CALL
ldap_tmplerr2string(int err)7637c478bd9Sstevel@tonic-gate ldap_tmplerr2string( int err )
7647c478bd9Sstevel@tonic-gate {
7657c478bd9Sstevel@tonic-gate 	static int init_flag = 0;
7667c478bd9Sstevel@tonic-gate 	int	i;
7677c478bd9Sstevel@tonic-gate 
7687c478bd9Sstevel@tonic-gate 	/* Multiple threads should be ok since they assign same strings */
7697c478bd9Sstevel@tonic-gate 	if (init_flag == 0) {
7707c478bd9Sstevel@tonic-gate 		ldap_tmplerrlist[0].e_reason =
7717c478bd9Sstevel@tonic-gate 			dgettext(TEXT_DOMAIN, "Bad template version");
7727c478bd9Sstevel@tonic-gate 		ldap_tmplerrlist[1].e_reason =
7737c478bd9Sstevel@tonic-gate 			dgettext(TEXT_DOMAIN, "Out of memory");
7747c478bd9Sstevel@tonic-gate 		ldap_tmplerrlist[2].e_reason =
7757c478bd9Sstevel@tonic-gate 			dgettext(TEXT_DOMAIN, "Bad template syntax");
7767c478bd9Sstevel@tonic-gate 		ldap_tmplerrlist[3].e_reason =
7777c478bd9Sstevel@tonic-gate 			dgettext(TEXT_DOMAIN, "File error reading template");
7787c478bd9Sstevel@tonic-gate 		init_flag = 1;
7797c478bd9Sstevel@tonic-gate 	}
7807c478bd9Sstevel@tonic-gate 
7817c478bd9Sstevel@tonic-gate 	for ( i = 0; ldap_tmplerrlist[i].e_code != -1; i++ ) {
7827c478bd9Sstevel@tonic-gate 		if ( err == ldap_tmplerrlist[i].e_code )
7837c478bd9Sstevel@tonic-gate 			return( ldap_tmplerrlist[i].e_reason );
7847c478bd9Sstevel@tonic-gate 	}
7857c478bd9Sstevel@tonic-gate 
7867c478bd9Sstevel@tonic-gate 	return(dgettext(TEXT_DOMAIN, "Unknown error") );
7877c478bd9Sstevel@tonic-gate }
788