X-Git-Url: https://git.sur5r.net/?a=blobdiff_plain;f=servers%2Fslapd%2Fat.c;h=854c81b3bca5f3ccfb28eb318874f31f062d8ea2;hb=294da7ed11a9a5721b15d0ed81682f123e20cbb0;hp=238b91f68443d23c19ef5f722b5540f6db0c1a20;hpb=69ba1a21bc6186fcb26e6f1e75226f9d4e69316a;p=openldap diff --git a/servers/slapd/at.c b/servers/slapd/at.c index 238b91f684..854c81b3bc 100644 --- a/servers/slapd/at.c +++ b/servers/slapd/at.c @@ -1,9 +1,18 @@ +/* at.c - routines for dealing with attribute types */ /* $OpenLDAP$ */ -/* - * Copyright 1998-2002 The OpenLDAP Foundation, All Rights Reserved. - * COPYING RESTRICTIONS APPLY, see COPYRIGHT file +/* This work is part of OpenLDAP Software . + * + * Copyright 1998-2008 The OpenLDAP Foundation. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted only as authorized by the OpenLDAP + * Public License. + * + * A copy of this license is available in the file LICENSE in the + * top-level directory of the distribution or, alternatively, at + * . */ -/* at.c - routines for dealing with attribute types */ #include "portable.h" @@ -15,20 +24,35 @@ #include #include -#include "ldap_pvt.h" #include "slap.h" -int is_at_syntax( - AttributeType *at, - const char *oid ) +const char * +at_syntax( + AttributeType *at ) { - for( ; at != NULL; at = at->sat_sup ) { - if( at->sat_syntax_oid ) { - return ( strcmp( at->sat_syntax_oid, oid ) == 0 ); + for ( ; at != NULL; at = at->sat_sup ) { + if ( at->sat_syntax_oid ) { + return at->sat_syntax_oid; } } + assert( 0 ); + + return NULL; +} + +int +is_at_syntax( + AttributeType *at, + const char *oid ) +{ + const char *syn_oid = at_syntax( at ); + + if ( syn_oid ) { + return strcmp( syn_oid, oid ) == 0; + } + return 0; } @@ -49,37 +73,41 @@ struct aindexrec { }; static Avlnode *attr_index = NULL; -static AttributeType *attr_list = NULL; +static Avlnode *attr_cache = NULL; +static LDAP_STAILQ_HEAD(ATList, AttributeType) attr_list + = LDAP_STAILQ_HEAD_INITIALIZER(attr_list); + +/* Last hardcoded attribute registered */ +AttributeType *at_sys_tail; + +int at_oc_cache; static int attr_index_cmp( - struct aindexrec *air1, - struct aindexrec *air2 -) + const void *v_air1, + const void *v_air2 ) { + const struct aindexrec *air1 = v_air1; + const struct aindexrec *air2 = v_air2; int i = air1->air_name.bv_len - air2->air_name.bv_len; - if (i) - return i; + if (i) return i; return (strcasecmp( air1->air_name.bv_val, air2->air_name.bv_val )); } static int attr_index_name_cmp( - struct berval *type, - struct aindexrec *air -) + const void *v_type, + const void *v_air ) { + const struct berval *type = v_type; + const struct aindexrec *air = v_air; int i = type->bv_len - air->air_name.bv_len; - if (i) - return i; - return (strncasecmp( type->bv_val, air->air_name.bv_val, - type->bv_len )); + if (i) return i; + return (strncasecmp( type->bv_val, air->air_name.bv_val, type->bv_len )); } AttributeType * -at_find( - const char *name -) +at_find( const char *name ) { struct berval bv; @@ -90,14 +118,25 @@ at_find( } AttributeType * -at_bvfind( - struct berval *name -) +at_bvfind( struct berval *name ) { struct aindexrec *air; - air = (struct aindexrec *) avl_find( attr_index, name, - (AVL_CMP) attr_index_name_cmp ); + if ( attr_cache ) { + air = avl_find( attr_cache, name, attr_index_name_cmp ); + if ( air ) return air->air_at; + } + + air = avl_find( attr_index, name, attr_index_name_cmp ); + + if ( air ) { + if ( air->air_at->sat_flags & SLAP_AT_DELETED ) { + air = NULL; + } else if (( slapMode & SLAP_TOOL_MODE ) && at_oc_cache ) { + avl_insert( &attr_cache, (caddr_t) air, + attr_index_cmp, avl_dup_error ); + } + } return air != NULL ? air->air_at : NULL; } @@ -105,8 +144,7 @@ at_bvfind( int at_append_to_list( AttributeType *sat, - AttributeType ***listp -) + AttributeType ***listp ) { AttributeType **list; AttributeType **list1; @@ -142,8 +180,7 @@ at_append_to_list( int at_delete_from_list( int pos, - AttributeType ***listp -) + AttributeType ***listp ) { AttributeType **list; AttributeType **list1; @@ -175,8 +212,7 @@ at_delete_from_list( int at_find_in_list( AttributeType *sat, - AttributeType **list -) + AttributeType **list ) { int i; @@ -191,29 +227,115 @@ at_find_in_list( return -1; } +static void +at_delete_names( AttributeType *at ) +{ + char **names = at->sat_names; + + while (*names) { + struct aindexrec tmpair, *air; + + ber_str2bv( *names, 0, 0, &tmpair.air_name ); + tmpair.air_at = at; + air = (struct aindexrec *)avl_delete( &attr_index, + (caddr_t)&tmpair, attr_index_cmp ); + assert( air != NULL ); + ldap_memfree( air ); + names++; + } +} + +/* Mark the attribute as deleted, remove from list, and remove all its + * names from the AVL tree. Leave the OID in the tree. + */ +void +at_delete( AttributeType *at ) +{ + at->sat_flags |= SLAP_AT_DELETED; + + LDAP_STAILQ_REMOVE(&attr_list, at, AttributeType, sat_next); + + at_delete_names( at ); +} + +static void +at_clean( AttributeType *a ) +{ + if ( a->sat_equality ) { + MatchingRule *mr; + + mr = mr_find( a->sat_equality->smr_oid ); + assert( mr != NULL ); + if ( mr != a->sat_equality ) { + ch_free( a->sat_equality ); + a->sat_equality = NULL; + } + } + + assert( a->sat_syntax != NULL ); + if ( a->sat_syntax != NULL ) { + Syntax *syn; + + syn = syn_find( a->sat_syntax->ssyn_oid ); + assert( syn != NULL ); + if ( syn != a->sat_syntax ) { + ch_free( a->sat_syntax ); + a->sat_syntax = NULL; + } + } + + if ( a->sat_oidmacro ) { + ldap_memfree( a->sat_oidmacro ); + a->sat_oidmacro = NULL; + } + if ( a->sat_subtypes ) { + ldap_memfree( a->sat_subtypes ); + a->sat_subtypes = NULL; + } +} + +static void +at_destroy_one( void *v ) +{ + struct aindexrec *air = v; + AttributeType *a = air->air_at; + + at_clean( a ); + ad_destroy(a->sat_ad); + ldap_pvt_thread_mutex_destroy(&a->sat_ad_mutex); + ldap_attributetype_free((LDAPAttributeType *)a); + ldap_memfree(air); +} + void at_destroy( void ) { - AttributeType *a, *n; - avl_free(attr_index, ldap_memfree); + AttributeType *a; + + while( !LDAP_STAILQ_EMPTY(&attr_list) ) { + a = LDAP_STAILQ_FIRST(&attr_list); + LDAP_STAILQ_REMOVE_HEAD(&attr_list, sat_next); - for (a=attr_list; a; a=n) { - n = a->sat_next; - if (a->sat_subtypes) ldap_memfree(a->sat_subtypes); - ad_destroy(a->sat_ad); - ldap_pvt_thread_mutex_destroy(&a->sat_ad_mutex); - ldap_attributetype_free((LDAPAttributeType *)a); + at_delete_names( a ); } - if ( slap_schema.si_at_undefined ) + + avl_free(attr_index, at_destroy_one); + + if ( slap_schema.si_at_undefined ) { ad_destroy(slap_schema.si_at_undefined->sat_ad); + } + + if ( slap_schema.si_at_proxied ) { + ad_destroy(slap_schema.si_at_proxied->sat_ad); + } } int at_start( AttributeType **at ) { - assert( at ); + assert( at != NULL ); - *at = attr_list; + *at = LDAP_STAILQ_FIRST(&attr_list); return (*at != NULL); } @@ -221,75 +343,206 @@ at_start( AttributeType **at ) int at_next( AttributeType **at ) { - assert( at ); + assert( at != NULL ); -#if 1 /* pedantic check */ +#if 0 /* pedantic check: don't use this */ { - AttributeType *tmp; + AttributeType *tmp = NULL; - for ( tmp = attr_list; tmp; tmp = tmp->sat_next ) { + LDAP_STAILQ_FOREACH(tmp,&attr_list,sat_next) { if ( tmp == *at ) { break; } } - assert( tmp ); + assert( tmp != NULL ); } #endif - *at = (*at)->sat_next; + *at = LDAP_STAILQ_NEXT(*at,sat_next); return (*at != NULL); } - +/* + * check whether the two attributeTypes actually __are__ identical, + * or rather inconsistent + */ +static int +at_check_dup( + AttributeType *sat, + AttributeType *new_sat ) +{ + if ( new_sat->sat_oid != NULL ) { + if ( sat->sat_oid == NULL ) { + return SLAP_SCHERR_ATTR_INCONSISTENT; + } + + if ( strcmp( sat->sat_oid, new_sat->sat_oid ) != 0 ) { + return SLAP_SCHERR_ATTR_INCONSISTENT; + } + + } else { + if ( sat->sat_oid != NULL ) { + return SLAP_SCHERR_ATTR_INCONSISTENT; + } + } + + if ( new_sat->sat_names ) { + int i; + + if ( sat->sat_names == NULL ) { + return SLAP_SCHERR_ATTR_INCONSISTENT; + } + + for ( i = 0; new_sat->sat_names[ i ]; i++ ) { + if ( sat->sat_names[ i ] == NULL ) { + return SLAP_SCHERR_ATTR_INCONSISTENT; + } + + if ( strcasecmp( sat->sat_names[ i ], + new_sat->sat_names[ i ] ) != 0 ) + { + return SLAP_SCHERR_ATTR_INCONSISTENT; + } + } + } else { + if ( sat->sat_names != NULL ) { + return SLAP_SCHERR_ATTR_INCONSISTENT; + } + } + + return SLAP_SCHERR_ATTR_DUP; +} + +static struct aindexrec *air_old; + +static int +at_dup_error( void *left, void *right ) +{ + air_old = left; + return -1; +} static int at_insert( - AttributeType *sat, - const char **err -) + AttributeType **rat, + AttributeType *prev, + const char **err ) { - AttributeType **atp; struct aindexrec *air; - char **names; - - atp = &attr_list; - while ( *atp != NULL ) { - atp = &(*atp)->sat_next; - } - *atp = sat; + char **names = NULL; + AttributeType *sat = *rat; if ( sat->sat_oid ) { air = (struct aindexrec *) ch_calloc( 1, sizeof(struct aindexrec) ); - air->air_name.bv_val = sat->sat_oid; - air->air_name.bv_len = strlen(sat->sat_oid); + ber_str2bv( sat->sat_oid, 0, 0, &air->air_name ); air->air_at = sat; + air_old = NULL; + if ( avl_insert( &attr_index, (caddr_t) air, - (AVL_CMP) attr_index_cmp, - (AVL_DUP) avl_dup_error ) ) { + attr_index_cmp, at_dup_error ) ) + { + AttributeType *old_sat; + int rc; + *err = sat->sat_oid; - ldap_memfree(air); - return SLAP_SCHERR_ATTR_DUP; + + assert( air_old != NULL ); + old_sat = air_old->air_at; + + /* replacing a deleted definition? */ + if ( old_sat->sat_flags & SLAP_AT_DELETED ) { + AttributeType tmp; + AttributeDescription *ad; + + /* Keep old oid, free new oid; + * Keep old ads, free new ads; + * Keep old ad_mutex, free new ad_mutex; + * Keep new everything else, free old + */ + tmp = *old_sat; + *old_sat = *sat; + old_sat->sat_oid = tmp.sat_oid; + tmp.sat_oid = sat->sat_oid; + old_sat->sat_ad = tmp.sat_ad; + tmp.sat_ad = sat->sat_ad; + old_sat->sat_ad_mutex = tmp.sat_ad_mutex; + tmp.sat_ad_mutex = sat->sat_ad_mutex; + *sat = tmp; + + /* Check for basic ad pointing at old cname */ + for ( ad = old_sat->sat_ad; ad; ad=ad->ad_next ) { + if ( ad->ad_cname.bv_val == sat->sat_cname.bv_val ) { + ad->ad_cname = old_sat->sat_cname; + break; + } + } + + at_clean( sat ); + at_destroy_one( air ); + + air = air_old; + sat = old_sat; + *rat = sat; + } else { + ldap_memfree( air ); + + rc = at_check_dup( old_sat, sat ); + + return rc; + } } /* FIX: temporal consistency check */ - at_bvfind(&air->air_name); + at_bvfind( &air->air_name ); } - if ( (names = sat->sat_names) ) { + names = sat->sat_names; + if ( names ) { while ( *names ) { air = (struct aindexrec *) ch_calloc( 1, sizeof(struct aindexrec) ); - air->air_name.bv_val = *names; - air->air_name.bv_len = strlen(*names); + ber_str2bv( *names, 0, 0, &air->air_name ); air->air_at = sat; if ( avl_insert( &attr_index, (caddr_t) air, - (AVL_CMP) attr_index_cmp, - (AVL_DUP) avl_dup_error ) ) { + attr_index_cmp, avl_dup_error ) ) + { + AttributeType *old_sat; + int rc; + *err = *names; + + old_sat = at_bvfind( &air->air_name ); + assert( old_sat != NULL ); + rc = at_check_dup( old_sat, sat ); + ldap_memfree(air); - return SLAP_SCHERR_ATTR_DUP; + + while ( names > sat->sat_names ) { + struct aindexrec tmpair; + + names--; + ber_str2bv( *names, 0, 0, &tmpair.air_name ); + tmpair.air_at = sat; + air = (struct aindexrec *)avl_delete( &attr_index, + (caddr_t)&tmpair, attr_index_cmp ); + assert( air != NULL ); + ldap_memfree( air ); + } + + if ( sat->sat_oid ) { + struct aindexrec tmpair; + + ber_str2bv( sat->sat_oid, 0, 0, &tmpair.air_name ); + tmpair.air_at = sat; + air = (struct aindexrec *)avl_delete( &attr_index, + (caddr_t)&tmpair, attr_index_cmp ); + assert( air != NULL ); + ldap_memfree( air ); + } + + return rc; } /* FIX: temporal consistency check */ at_bvfind(&air->air_name); @@ -297,24 +550,55 @@ at_insert( } } + if ( sat->sat_oid ) { + slap_ad_undef_promote( sat->sat_oid, sat ); + } + + names = sat->sat_names; + if ( names ) { + while ( *names ) { + slap_ad_undef_promote( *names, sat ); + names++; + } + } + + if ( sat->sat_flags & SLAP_AT_HARDCODE ) { + prev = at_sys_tail; + at_sys_tail = sat; + } + if ( prev ) { + LDAP_STAILQ_INSERT_AFTER( &attr_list, prev, sat, sat_next ); + } else { + LDAP_STAILQ_INSERT_TAIL( &attr_list, sat, sat_next ); + } + return 0; } int at_add( - LDAPAttributeType *at, - const char **err -) + LDAPAttributeType *at, + int user, + AttributeType **rsat, + AttributeType *prev, + const char **err ) { - AttributeType *sat; - MatchingRule *mr; - Syntax *syn; + AttributeType *sat = NULL; + MatchingRule *mr = NULL; + Syntax *syn = NULL; int i; - int code; - char *cname; - char *oid; + int code = LDAP_SUCCESS; + char *cname = NULL; + char *oidm = NULL; + + if ( !at->at_oid ) { + *err = ""; + return SLAP_SCHERR_ATTR_INCOMPLETE; + } if ( !OID_LEADCHAR( at->at_oid[0] )) { + char *oid; + /* Expand OID macros */ oid = oidm_find( at->at_oid ); if ( !oid ) { @@ -322,23 +606,25 @@ at_add( return SLAP_SCHERR_OIDM; } if ( oid != at->at_oid ) { - ldap_memfree( at->at_oid ); + oidm = at->at_oid; at->at_oid = oid; } } if ( at->at_syntax_oid && !OID_LEADCHAR( at->at_syntax_oid[0] )) { + char *oid; + /* Expand OID macros */ oid = oidm_find( at->at_syntax_oid ); if ( !oid ) { *err = at->at_syntax_oid; - return SLAP_SCHERR_OIDM; + code = SLAP_SCHERR_OIDM; + goto error_return; } if ( oid != at->at_syntax_oid ) { ldap_memfree( at->at_syntax_oid ); at->at_syntax_oid = oid; } - } if ( at->at_names && at->at_names[0] ) { @@ -347,36 +633,37 @@ at_add( for( i=0; at->at_names[i]; i++ ) { if( !slap_valid_descr( at->at_names[i] ) ) { *err = at->at_names[i]; - return SLAP_SCHERR_BAD_DESCR; + code = SLAP_SCHERR_BAD_DESCR; + goto error_return; } } cname = at->at_names[0]; - } else if ( at->at_oid ) { + } else { cname = at->at_oid; - } else { - *err = ""; - return SLAP_SCHERR_ATTR_INCOMPLETE; } *err = cname; if ( !at->at_usage && at->at_no_user_mod ) { /* user attribute must be modifable */ - return SLAP_SCHERR_ATTR_BAD_USAGE; + code = SLAP_SCHERR_ATTR_BAD_USAGE; + goto error_return; } if ( at->at_collective ) { if( at->at_usage ) { /* collective attributes cannot be operational */ - return SLAP_SCHERR_ATTR_BAD_USAGE; + code = SLAP_SCHERR_ATTR_BAD_USAGE; + goto error_return; } if( at->at_single_value ) { /* collective attributes cannot be single-valued */ - return SLAP_SCHERR_ATTR_BAD_USAGE; + code = SLAP_SCHERR_ATTR_BAD_USAGE; + goto error_return; } } @@ -385,6 +672,7 @@ at_add( sat->sat_cname.bv_val = cname; sat->sat_cname.bv_len = strlen( cname ); + sat->sat_oidmacro = oidm; ldap_pvt_thread_mutex_init(&sat->sat_ad_mutex); if ( at->at_sup_oid ) { @@ -392,23 +680,33 @@ at_add( if ( supsat == NULL ) { *err = at->at_sup_oid; - return SLAP_SCHERR_ATTR_NOT_FOUND; + code = SLAP_SCHERR_ATTR_NOT_FOUND; + goto error_return; } sat->sat_sup = supsat; if ( at_append_to_list(sat, &supsat->sat_subtypes) ) { - return SLAP_SCHERR_OUTOFMEM; + code = SLAP_SCHERR_OUTOFMEM; + goto error_return; } if ( sat->sat_usage != supsat->sat_usage ) { /* subtypes must have same usage as their SUP */ - return SLAP_SCHERR_ATTR_BAD_USAGE; + code = SLAP_SCHERR_ATTR_BAD_USAGE; + goto error_return; + } + + if ( supsat->sat_obsolete && !sat->sat_obsolete ) { + /* subtypes must be obsolete if super is */ + code = SLAP_SCHERR_ATTR_BAD_SUP; + goto error_return; } if ( sat->sat_flags & SLAP_AT_FINAL ) { /* cannot subtype a "final" attribute type */ - return SLAP_SCHERR_ATTR_BAD_SUP; + code = SLAP_SCHERR_ATTR_BAD_SUP; + goto error_return; } } @@ -425,21 +723,50 @@ at_add( sat->sat_substr = sat->sat_sup->sat_substr; } + /* + * check for X-ORDERED attributes + */ + if ( sat->sat_extensions ) { + for (i=0; sat->sat_extensions[i]; i++) { + if (!strcasecmp( sat->sat_extensions[i]->lsei_name, + "X-ORDERED" ) && sat->sat_extensions[i]->lsei_values ) { + if ( !strcasecmp( sat->sat_extensions[i]->lsei_values[0], + "VALUES" )) { + sat->sat_flags |= SLAP_AT_ORDERED_VAL; + break; + } else if ( !strcasecmp( sat->sat_extensions[i]->lsei_values[0], + "SIBLINGS" )) { + sat->sat_flags |= SLAP_AT_ORDERED_SIB; + break; + } + } + } + } + + if ( !user ) + sat->sat_flags |= SLAP_AT_HARDCODE; + if ( at->at_syntax_oid ) { syn = syn_find(sat->sat_syntax_oid); if ( syn == NULL ) { *err = sat->sat_syntax_oid; - return SLAP_SCHERR_SYN_NOT_FOUND; + code = SLAP_SCHERR_SYN_NOT_FOUND; + goto error_return; } - if( sat->sat_syntax != NULL && sat->sat_syntax != syn ) { - return SLAP_SCHERR_ATTR_BAD_SUP; + if ( sat->sat_syntax != NULL && sat->sat_syntax != syn ) { + /* BEWARE: no loop detection! */ + if ( syn_is_sup( sat->sat_syntax, syn ) ) { + code = SLAP_SCHERR_ATTR_BAD_SUP; + goto error_return; + } } sat->sat_syntax = syn; } else if ( sat->sat_syntax == NULL ) { - return SLAP_SCHERR_ATTR_INCOMPLETE; + code = SLAP_SCHERR_ATTR_INCOMPLETE; + goto error_return; } if ( sat->sat_equality_oid ) { @@ -447,18 +774,21 @@ at_add( if( mr == NULL ) { *err = sat->sat_equality_oid; - return SLAP_SCHERR_MR_NOT_FOUND; + code = SLAP_SCHERR_MR_NOT_FOUND; + goto error_return; } if(( mr->smr_usage & SLAP_MR_EQUALITY ) != SLAP_MR_EQUALITY ) { *err = sat->sat_equality_oid; - return SLAP_SCHERR_ATTR_BAD_MR; + code = SLAP_SCHERR_ATTR_BAD_MR; + goto error_return; } if( sat->sat_syntax != mr->smr_syntax ) { if( mr->smr_compat_syntaxes == NULL ) { *err = sat->sat_equality_oid; - return SLAP_SCHERR_ATTR_BAD_MR; + code = SLAP_SCHERR_ATTR_BAD_MR; + goto error_return; } for(i=0; mr->smr_compat_syntaxes[i]; i++) { @@ -470,7 +800,8 @@ at_add( if( i >= 0 ) { *err = sat->sat_equality_oid; - return SLAP_SCHERR_ATTR_BAD_MR; + code = SLAP_SCHERR_ATTR_BAD_MR; + goto error_return; } } @@ -479,22 +810,31 @@ at_add( } if ( sat->sat_ordering_oid ) { + if( !sat->sat_equality ) { + *err = sat->sat_ordering_oid; + code = SLAP_SCHERR_ATTR_BAD_MR; + goto error_return; + } + mr = mr_find(sat->sat_ordering_oid); if( mr == NULL ) { *err = sat->sat_ordering_oid; - return SLAP_SCHERR_MR_NOT_FOUND; + code = SLAP_SCHERR_MR_NOT_FOUND; + goto error_return; } if(( mr->smr_usage & SLAP_MR_ORDERING ) != SLAP_MR_ORDERING ) { *err = sat->sat_ordering_oid; - return SLAP_SCHERR_ATTR_BAD_MR; + code = SLAP_SCHERR_ATTR_BAD_MR; + goto error_return; } if( sat->sat_syntax != mr->smr_syntax ) { if( mr->smr_compat_syntaxes == NULL ) { *err = sat->sat_ordering_oid; - return SLAP_SCHERR_ATTR_BAD_MR; + code = SLAP_SCHERR_ATTR_BAD_MR; + goto error_return; } for(i=0; mr->smr_compat_syntaxes[i]; i++) { @@ -506,7 +846,8 @@ at_add( if( i >= 0 ) { *err = sat->sat_ordering_oid; - return SLAP_SCHERR_ATTR_BAD_MR; + code = SLAP_SCHERR_ATTR_BAD_MR; + goto error_return; } } @@ -514,29 +855,36 @@ at_add( } if ( sat->sat_substr_oid ) { + if( !sat->sat_equality ) { + *err = sat->sat_substr_oid; + code = SLAP_SCHERR_ATTR_BAD_MR; + goto error_return; + } + mr = mr_find(sat->sat_substr_oid); if( mr == NULL ) { *err = sat->sat_substr_oid; - return SLAP_SCHERR_MR_NOT_FOUND; + code = SLAP_SCHERR_MR_NOT_FOUND; + goto error_return; } if(( mr->smr_usage & SLAP_MR_SUBSTR ) != SLAP_MR_SUBSTR ) { *err = sat->sat_substr_oid; - return SLAP_SCHERR_ATTR_BAD_MR; + code = SLAP_SCHERR_ATTR_BAD_MR; + goto error_return; } - /* due to funky LDAP builtin substring rules, we + /* due to funky LDAP builtin substring rules, * we check against the equality rule assertion * syntax and compat syntaxes instead of those * associated with the substrings rule. */ - if( sat->sat_equality && - sat->sat_syntax != sat->sat_equality->smr_syntax ) - { + if( sat->sat_syntax != sat->sat_equality->smr_syntax ) { if( sat->sat_equality->smr_compat_syntaxes == NULL ) { *err = sat->sat_substr_oid; - return SLAP_SCHERR_ATTR_BAD_MR; + code = SLAP_SCHERR_ATTR_BAD_MR; + goto error_return; } for(i=0; sat->sat_equality->smr_compat_syntaxes[i]; i++) { @@ -550,22 +898,40 @@ at_add( if( i >= 0 ) { *err = sat->sat_substr_oid; - return SLAP_SCHERR_ATTR_BAD_MR; + code = SLAP_SCHERR_ATTR_BAD_MR; + goto error_return; } } sat->sat_substr = mr; } - code = at_insert(sat,err); + code = at_insert( &sat, prev, err ); + if ( code != 0 ) { +error_return:; + if ( sat ) { + ldap_pvt_thread_mutex_destroy( &sat->sat_ad_mutex ); + ch_free( sat ); + } + + if ( oidm ) { + SLAP_FREE( at->at_oid ); + at->at_oid = oidm; + } + + } else if ( rsat ) { + *rsat = sat; + } + return code; } #ifdef LDAP_DEBUG +#ifdef SLAPD_UNUSED static int -at_index_printnode( struct aindexrec *air ) +at_index_printnode( void *v_air, void *ignore ) { - + struct aindexrec *air = v_air; printf("%s = %s\n", air->air_name.bv_val, ldap_attributetype2str(&air->air_at->sat_atype) ); @@ -576,36 +942,133 @@ static void at_index_print( void ) { printf("Printing attribute type index:\n"); - (void) avl_apply( attr_index, (AVL_APPLY) at_index_printnode, - 0, -1, AVL_INORDER ); + (void) avl_apply( attr_index, at_index_printnode, 0, -1, AVL_INORDER ); } #endif +#endif + +void +at_unparse( BerVarray *res, AttributeType *start, AttributeType *end, int sys ) +{ + AttributeType *at; + int i, num; + struct berval bv, *bva = NULL, idx; + char ibuf[32]; + + if ( !start ) + start = LDAP_STAILQ_FIRST( &attr_list ); + + /* count the result size */ + i = 0; + for ( at=start; at; at=LDAP_STAILQ_NEXT(at, sat_next)) { + if ( sys && !(at->sat_flags & SLAP_AT_HARDCODE)) break; + i++; + if ( at == end ) break; + } + if (!i) return; + + num = i; + bva = ch_malloc( (num+1) * sizeof(struct berval) ); + BER_BVZERO( bva ); + idx.bv_val = ibuf; + if ( sys ) { + idx.bv_len = 0; + ibuf[0] = '\0'; + } + i = 0; + for ( at=start; at; at=LDAP_STAILQ_NEXT(at, sat_next)) { + LDAPAttributeType lat, *latp; + if ( sys && !(at->sat_flags & SLAP_AT_HARDCODE)) break; + if ( at->sat_oidmacro ) { + lat = at->sat_atype; + lat.at_oid = at->sat_oidmacro; + latp = ⪫ + } else { + latp = &at->sat_atype; + } + if ( ldap_attributetype2bv( latp, &bv ) == NULL ) { + ber_bvarray_free( bva ); + } + if ( !sys ) { + idx.bv_len = sprintf(idx.bv_val, "{%d}", i); + } + bva[i].bv_len = idx.bv_len + bv.bv_len; + bva[i].bv_val = ch_malloc( bva[i].bv_len + 1 ); + strcpy( bva[i].bv_val, ibuf ); + strcpy( bva[i].bv_val + idx.bv_len, bv.bv_val ); + i++; + bva[i].bv_val = NULL; + ldap_memfree( bv.bv_val ); + if ( at == end ) break; + } + *res = bva; +} -#if defined( SLAPD_SCHEMA_DN ) int at_schema_info( Entry *e ) { - struct berval vals[2]; - AttributeType *at; - AttributeDescription *ad_attributeTypes = slap_schema.si_ad_attributeTypes; + AttributeType *at; + struct berval val; + struct berval nval; - vals[1].bv_val = NULL; + LDAP_STAILQ_FOREACH(at,&attr_list,sat_next) { + if( at->sat_flags & SLAP_AT_HIDE ) continue; - for ( at = attr_list; at; at = at->sat_next ) { - if ( ldap_attributetype2bv( &at->sat_atype, vals ) == NULL ) { + if ( ldap_attributetype2bv( &at->sat_atype, &val ) == NULL ) { return -1; } - if( at->sat_flags & SLAP_AT_HIDE ) continue; + ber_str2bv( at->sat_oid, 0, 0, &nval ); -#if 0 - Debug( LDAP_DEBUG_TRACE, "Merging at [%ld] %s\n", - (long) vals[0].bv_len, vals[0].bv_val, 0 ); -#endif - attr_merge( e, ad_attributeTypes, vals ); - ldap_memfree( vals[0].bv_val ); + if( attr_merge_one( e, ad_attributeTypes, &val, &nval ) ) + { + return -1; + } + ldap_memfree( val.bv_val ); } return 0; } -#endif + +int +register_at( char *def, AttributeDescription **rad, int dupok ) +{ + LDAPAttributeType *at; + int code, freeit = 0; + const char *err; + AttributeDescription *ad = NULL; + + at = ldap_str2attributetype( def, &code, &err, LDAP_SCHEMA_ALLOW_ALL ); + if ( !at ) { + Debug( LDAP_DEBUG_ANY, + "register_at: AttributeType \"%s\": %s, %s\n", + def, ldap_scherr2str(code), err ); + return code; + } + + code = at_add( at, 0, NULL, NULL, &err ); + if ( code ) { + if ( code == SLAP_SCHERR_ATTR_DUP && dupok ) { + freeit = 1; + + } else { + ldap_attributetype_free( at ); + Debug( LDAP_DEBUG_ANY, + "register_at: AttributeType \"%s\": %s, %s\n", + def, scherr2str(code), err ); + return code; + } + } + code = slap_str2ad( at->at_names[0], &ad, &err ); + if ( freeit || code ) { + ldap_attributetype_free( at ); + } else { + ldap_memfree( at ); + } + if ( code ) { + Debug( LDAP_DEBUG_ANY, "register_at: AttributeType \"%s\": %s\n", + def, err, 0 ); + } + if ( rad ) *rad = ad; + return code; +}