#include "ldap_pvt.h"
#include "slap.h"
-AttributeDescription *ad_dup(
- AttributeDescription *desc )
-{
- AttributeDescription *ad;
-
- if( desc == NULL ) {
- return NULL;
- }
-
- ad = (AttributeDescription *) ch_malloc( sizeof(AttributeDescription) );
-
- *ad = *desc;
-
- if( ad->ad_cname != NULL ) {
- ad->ad_cname = ber_bvdup( ad->ad_cname );
- }
-
- if( ad->ad_lang != NULL ) {
- ad->ad_lang = ch_strdup( ad->ad_lang );
- }
-
- return ad;
-}
-
-void
-ad_free( AttributeDescription *ad, int freeit )
-{
- if( ad == NULL ) return;
-
- if( ad->ad_cname != NULL ) {
- ber_bvfree( ad->ad_cname );
- }
-
- free( ad->ad_lang );
-
- if( freeit ) free( ad );
-}
-
static int ad_keystring(
struct berval *bv )
{
return 0;
}
+void ad_destroy( void *in )
+{
+ AttributeDescription *ad = in, *n;
+
+ for (;ad;ad = n) {
+ n = ad->ad_next;
+ ldap_memfree(ad);
+ }
+}
+
+/* Is there an AttributeDescription for this type that uses this language? */
+AttributeDescription * ad_find_lang(
+ AttributeType *type,
+ struct berval *lang )
+{
+ AttributeDescription *ad;
+
+ ldap_pvt_thread_mutex_lock( &type->sat_ad_mutex );
+ for (ad = type->sat_ad; ad; ad=ad->ad_next)
+ {
+ if (ad->ad_lang.bv_len == lang->bv_len &&
+ !strcasecmp(ad->ad_lang.bv_val, lang->bv_val))
+ break;
+ }
+ ldap_pvt_thread_mutex_unlock( &type->sat_ad_mutex );
+ return ad;
+}
+
int slap_str2ad(
const char *str,
AttributeDescription **ad,
{
int rtn = LDAP_UNDEFINED_TYPE;
int i;
- AttributeDescription desc;
- char **tokens;
+ AttributeDescription desc, *d2;
+ char *name, *options;
assert( ad != NULL );
assert( *ad == NULL ); /* temporary */
return rtn;
}
- tokens = str2charray( bv->bv_val, ";");
-
- if( tokens == NULL || *tokens == NULL ) {
- *text = "no attribute type";
- goto done;
- }
-
- desc.ad_type = at_find( *tokens );
-
+ /* find valid base attribute type; parse in place */
+ name = bv->bv_val;
+ options = strchr(name, ';');
+ if (options != NULL) *options = '\0';
+ desc.ad_type = at_find( name );
+ if (options != NULL) *options = ';';
if( desc.ad_type == NULL ) {
*text = "attribute type undefined";
- goto done;
+ return rtn;
}
- desc.ad_flags = SLAP_DESC_NONE;
- desc.ad_lang = NULL;
+ if (options != NULL) {
+ desc.ad_cname.bv_len = options - name;
+ } else {
+ desc.ad_cname.bv_len = bv->bv_len;
+ }
- for( i=1; tokens[i] != NULL; i++ ) {
- if( strcasecmp( tokens[i], "binary" ) == 0 ) {
+ desc.ad_flags = SLAP_DESC_NONE;
+ desc.ad_lang.bv_len = 0;
+ desc.ad_lang.bv_val = NULL;
+
+ /* parse options in place */
+ for( ; options != NULL; ) {
+ name = options+1;
+ options = strchr( name, ';' );
+ if ( options != NULL )
+ i = options - name;
+ else
+ i = strlen(name);
+
+ if( i == sizeof("binary")-1 && strncasecmp( name, "binary", i) == 0 ) {
if( slap_ad_is_binary( &desc ) ) {
*text = "option \"binary\" specified multiple times";
goto done;
desc.ad_flags |= SLAP_DESC_BINARY;
- } else if ( strncasecmp( tokens[i], "lang-",
- sizeof("lang-")-1 ) == 0 && tokens[i][sizeof("lang-")-1] )
+ } else if ( i >= sizeof("lang-") && strncasecmp( name, "lang-",
+ sizeof("lang-")-1 ) == 0)
{
- if( desc.ad_lang != NULL ) {
+ if( desc.ad_lang.bv_len != 0 ) {
*text = "multiple language tag options specified";
goto done;
}
- desc.ad_lang = ch_strdup( tokens[i] );
-
- /* normalize to all lower case, it's easy */
- ldap_pvt_str2lower( desc.ad_lang );
-
+ desc.ad_lang.bv_val = name;
+ desc.ad_lang.bv_len = i;
} else {
*text = "unrecognized option";
goto done;
}
}
- desc.ad_cname = ch_malloc( sizeof( struct berval ) );
-
- desc.ad_cname->bv_len = strlen( desc.ad_type->sat_cname );
- if( slap_ad_is_binary( &desc ) ) {
- desc.ad_cname->bv_len += sizeof("binary");
+ ldap_pvt_thread_mutex_lock( &desc.ad_type->sat_ad_mutex );
+ /* see if a matching description is already cached */
+ for (d2 = desc.ad_type->sat_ad; d2; d2=d2->ad_next) {
+ if (d2->ad_flags != desc.ad_flags)
+ continue;
+ if (d2->ad_lang.bv_len != desc.ad_lang.bv_len)
+ continue;
+ if (d2->ad_lang.bv_len == 0)
+ break;
+ if (strncasecmp(d2->ad_lang.bv_val, desc.ad_lang.bv_val,
+ desc.ad_lang.bv_len) == 0)
+ break;
}
- if( desc.ad_lang != NULL ) {
- desc.ad_cname->bv_len += 1 + strlen( desc.ad_lang );
- }
-
- desc.ad_cname->bv_val = ch_malloc( desc.ad_cname->bv_len + 1 );
- strcpy( desc.ad_cname->bv_val, desc.ad_type->sat_cname );
- if( slap_ad_is_binary( &desc ) ) {
- strcat( desc.ad_cname->bv_val, ";binary" );
- }
+ /* Not found, add new one */
+ while (d2 == NULL) {
+ int dlen = 0;
+
+ /* Allocate a single contiguous block. If there are no
+ * options, we just need space for the AttrDesc structure.
+ * Otherwise, we need to tack on the full name length +
+ * options length.
+ */
+ i = sizeof(AttributeDescription);
+ if (desc.ad_lang.bv_len || desc.ad_flags != SLAP_DESC_NONE) {
+ if (desc.ad_lang.bv_len)
+ dlen = desc.ad_lang.bv_len+1;
+ dlen += desc.ad_type->sat_cname.bv_len+1;
+ if( slap_ad_is_binary( &desc ) ) {
+ dlen += sizeof("binary");
+ }
+ }
- if( desc.ad_lang != NULL ) {
- strcat( desc.ad_cname->bv_val, ";" );
- strcat( desc.ad_cname->bv_val, desc.ad_lang );
+ d2 = ch_malloc(i + dlen);
+ d2->ad_type = desc.ad_type;
+ d2->ad_flags = desc.ad_flags;
+ d2->ad_cname.bv_len = desc.ad_cname.bv_len;
+ d2->ad_lang.bv_len = desc.ad_lang.bv_len;
+ if (dlen == 0) {
+ d2->ad_cname.bv_val = d2->ad_type->sat_cname.bv_val;
+ d2->ad_lang.bv_val = NULL;
+ } else {
+ d2->ad_cname.bv_val = (char *)(d2+1);
+ strcpy(d2->ad_cname.bv_val, d2->ad_type->sat_cname.bv_val);
+ if( slap_ad_is_binary( &desc ) ) {
+ strcpy(d2->ad_cname.bv_val+d2->ad_cname.bv_len,
+ ";binary");
+ d2->ad_cname.bv_len += sizeof("binary");
+ }
+ if( d2->ad_lang.bv_len ) {
+ d2->ad_cname.bv_val[d2->ad_cname.bv_len++]=';';
+ d2->ad_lang.bv_val = d2->ad_cname.bv_val+
+ d2->ad_cname.bv_len;
+ strncpy(d2->ad_lang.bv_val,desc.ad_lang.bv_val,
+ d2->ad_lang.bv_len);
+ d2->ad_lang.bv_val[d2->ad_lang.bv_len] = '\0';
+ ldap_pvt_str2lower(d2->ad_lang.bv_val);
+ d2->ad_cname.bv_len += d2->ad_lang.bv_len;
+ }
+ }
+ /* Add new desc to list. We always want the bare Desc with
+ * no options to stay at the head of the list, assuming
+ * that one will be used most frequently.
+ */
+ if (desc.ad_type->sat_ad == NULL || dlen == 0) {
+ d2->ad_next = desc.ad_type->sat_ad;
+ desc.ad_type->sat_ad = d2;
+ } else {
+ d2->ad_next = desc.ad_type->sat_ad->ad_next;
+ desc.ad_type->sat_ad->ad_next = d2;
+ }
}
+ ldap_pvt_thread_mutex_unlock( &desc.ad_type->sat_ad_mutex );
if( *ad == NULL ) {
- *ad = ch_malloc( sizeof( AttributeDescription ) );
+ *ad = d2;
+ } else {
+ **ad = *d2;
}
- **ad = desc;
-
rtn = LDAP_SUCCESS;
done:
- charray_free( tokens );
return rtn;
}
return 0;
}
- if( super->ad_lang != NULL && ( sub->ad_lang == NULL
- || strcasecmp( super->ad_lang, sub->ad_lang )))
+ if( super->ad_lang.bv_len && (sub->ad_lang.bv_len !=
+ super->ad_lang.bv_len || strcmp( super->ad_lang.bv_val,
+ sub->ad_lang.bv_val)))
{
return 0;
}
rc = is_ad_subtype( desc, ad );
- ad_free( ad, 1 );
-
if( rc ) return 1;
}
AttributeDescription **ad,
const char **text )
{
- AttributeDescription desc;
+ AttributeDescription *desc;
assert( ad != NULL );
- assert( *ad == NULL ); /* temporary */
if( bv == NULL || bv->bv_len == 0 ) {
*text = "empty attribute description";
return LDAP_UNDEFINED_TYPE;
}
- desc.ad_type = slap_schema.si_at_undefined;
- desc.ad_flags = SLAP_DESC_NONE;
- desc.ad_lang = NULL;
+ for (desc = slap_schema.si_at_undefined->sat_ad; desc;
+ desc=desc->ad_next)
+ if (desc->ad_cname.bv_len == bv->bv_len &&
+ !strcasecmp(desc->ad_cname.bv_val, bv->bv_val))
+ break;
+
+ if (!desc) {
+ desc = ch_malloc(sizeof(AttributeDescription) +
+ bv->bv_len + 1);
+
+ desc->ad_flags = SLAP_DESC_NONE;
+ desc->ad_lang.bv_val = NULL;
+ desc->ad_lang.bv_len = 0;
- desc.ad_cname = ber_bvdup( bv );
+ desc->ad_cname.bv_len = bv->bv_len;
+ desc->ad_cname.bv_val = (char *)(desc+1);
+ strcpy(desc->ad_cname.bv_val, bv->bv_val);
- /* canoncial to upper case */
- ldap_pvt_str2upper( bv->bv_val );
+ /* canonical to upper case */
+ ldap_pvt_str2upper( desc->ad_cname.bv_val );
- if( *ad == NULL ) {
- *ad = ch_malloc( sizeof( AttributeDescription ) );
+ desc->ad_type = slap_schema.si_at_undefined;
+ desc->ad_next = desc->ad_type->sat_ad;
+ desc->ad_type->sat_ad = desc;
}
- **ad = desc;
+ if (!*ad)
+ *ad = desc;
+ else
+ **ad = *desc;
return LDAP_SUCCESS;
}
-