/* operational.c - bdb backend operational attributes function */ /* $OpenLDAP$ */ /* This work is part of OpenLDAP Software . * * Copyright 2000-2004 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 * . */ #include "portable.h" #include #include #include #include "slap.h" #include "back-bdb.h" /* * sets *hasSubordinates to LDAP_COMPARE_TRUE/LDAP_COMPARE_FALSE * if the entry has children or not. */ int bdb_hasSubordinates( Operation *op, Entry *e, int *hasSubordinates ) { int rc; assert( e ); /* NOTE: this should never happen, but it actually happens * when using back-relay; until we find a better way to * preserve entry's private information while rewriting it, * let's disable the hasSubordinate feature for back-relay. */ if ( BEI( e ) == NULL ) { return LDAP_OTHER; } retry: /* FIXME: we can no longer assume the entry's e_private * field is correctly populated; so we need to reacquire * it with reader lock */ rc = bdb_cache_children( op, NULL, e ); switch( rc ) { case DB_LOCK_DEADLOCK: case DB_LOCK_NOTGRANTED: ldap_pvt_thread_yield(); goto retry; case 0: *hasSubordinates = LDAP_COMPARE_TRUE; break; case DB_NOTFOUND: *hasSubordinates = LDAP_COMPARE_FALSE; rc = LDAP_SUCCESS; break; default: Debug(LDAP_DEBUG_ARGS, "<=- " LDAP_XSTRING(bdb_hasSubordinates) ": has_children failed: %s (%d)\n", db_strerror(rc), rc, 0 ); rc = LDAP_OTHER; } return rc; } /* * sets the supported operational attributes (if required) */ int bdb_operational( Operation *op, SlapReply *rs ) { Attribute **ap; assert( rs->sr_entry ); for ( ap = &rs->sr_operational_attrs; *ap; ap = &(*ap)->a_next ) /* just count */ ; if ( SLAP_OPATTRS( rs->sr_attr_flags ) || ad_inlist( slap_schema.si_ad_hasSubordinates, rs->sr_attrs ) ) { int hasSubordinates; rs->sr_err = bdb_hasSubordinates( op, rs->sr_entry, &hasSubordinates ); if ( rs->sr_err == LDAP_SUCCESS ) { *ap = slap_operational_hasSubordinate( hasSubordinates == LDAP_COMPARE_TRUE ); assert( *ap ); ap = &(*ap)->a_next; } } return rs->sr_err; }