/* cache.c - routines to maintain an in-core cache of entries */ /* $OpenLDAP$ */ /* This work is part of OpenLDAP Software . * * Copyright 2001-2022 The OpenLDAP Foundation. * Portions Copyright 2001-2003 Pierangelo Masarati. * 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 file LICENSE in the * top-level directory of the distribution or, alternatively, at * . */ /* ACKNOWLEDGEMENTS: * This work was initially developed by Pierangelo Masarati for inclusion * in OpenLDAP Software. */ #include "portable.h" #include #include "ac/string.h" #include "slap.h" #include "back-monitor.h" /* * The cache maps DNs to Entries. * Each entry, on turn, holds the list of its children in the e_private field. * This is used by search operation to perform onelevel and subtree candidate * selection. */ typedef struct monitor_cache_t { struct berval mc_ndn; Entry *mc_e; } monitor_cache_t; /* * compares entries based on the dn */ int monitor_cache_cmp( const void *c1, const void *c2 ) { monitor_cache_t *cc1 = ( monitor_cache_t * )c1; monitor_cache_t *cc2 = ( monitor_cache_t * )c2; /* * case sensitive, because the dn MUST be normalized */ return ber_bvcmp( &cc1->mc_ndn, &cc2->mc_ndn ); } /* * checks for duplicate entries */ int monitor_cache_dup( void *c1, void *c2 ) { monitor_cache_t *cc1 = ( monitor_cache_t * )c1; monitor_cache_t *cc2 = ( monitor_cache_t * )c2; /* * case sensitive, because the dn MUST be normalized */ return ber_bvcmp( &cc1->mc_ndn, &cc2->mc_ndn ) == 0 ? -1 : 0; } /* * adds an entry to the cache and inits the mutex */ int monitor_cache_add( monitor_info_t *mi, Entry *e, Entry *parent ) { monitor_cache_t tmp_mc, *mc, *pmc = NULL; Entry **ep = NULL, *prev = NULL; int rc = -1; assert( mi != NULL ); assert( e != NULL ); dnParent( &e->e_nname, &tmp_mc.mc_ndn ); mc = ( monitor_cache_t * )ch_malloc( sizeof( monitor_cache_t ) ); mc->mc_ndn = e->e_nname; mc->mc_e = e; if ( parent ) { /* Shortcut, but follow lock order as a fallback */ if ( ldap_pvt_thread_mutex_trylock( &mi->mi_cache_lock ) ) { monitor_cache_release( mi, parent ); ldap_pvt_thread_mutex_lock( &mi->mi_cache_lock ); monitor_cache_lock( parent ); } } else { ldap_pvt_thread_mutex_lock( &mi->mi_cache_lock ); } /* Allow database root be added */ if ( parent == NULL && mi->mi_cache != NULL ) { pmc = ldap_avl_find( mi->mi_cache, &tmp_mc, monitor_cache_cmp ); if ( pmc == NULL ) { goto done; } parent = pmc->mc_e; monitor_cache_lock( parent ); } rc = ldap_avl_insert( &mi->mi_cache, mc, monitor_cache_cmp, monitor_cache_dup ); if ( rc != LDAP_SUCCESS ) { goto done; } if ( parent != NULL ) { monitor_entry_t *mp = parent->e_private; if ( mp->mp_children ) { monitor_entry_t *tail; monitor_cache_lock( mp->mp_last ); tail = mp->mp_last->e_private; tail->mp_next = e; monitor_cache_release( mi, mp->mp_last ); mp->mp_last = e; } else { mp->mp_children = mp->mp_last = e; } } done: if ( pmc != NULL ) { monitor_cache_release( mi, parent ); } ldap_pvt_thread_mutex_unlock( &mi->mi_cache_lock ); if ( rc != LDAP_SUCCESS ) { ch_free( mc ); } return rc; } /* * locks the entry (no r/w) */ int monitor_cache_lock( Entry *e ) { monitor_entry_t *mp; assert( e != NULL ); assert( e->e_private != NULL ); mp = ( monitor_entry_t * )e->e_private; ldap_pvt_thread_mutex_lock( &mp->mp_mutex ); return( 0 ); } /* * tries to lock the entry (no r/w) */ int monitor_cache_trylock( Entry *e ) { monitor_entry_t *mp; assert( e != NULL ); assert( e->e_private != NULL ); mp = ( monitor_entry_t * )e->e_private; return ldap_pvt_thread_mutex_trylock( &mp->mp_mutex ); } /* * gets an entry from the cache based on the normalized dn * with mutex locked */ int monitor_cache_get( monitor_info_t *mi, struct berval *ndn, Entry **ep ) { monitor_cache_t tmp_mc, *mc; assert( mi != NULL ); assert( ndn != NULL ); assert( ep != NULL ); *ep = NULL; tmp_mc.mc_ndn = *ndn; ldap_pvt_thread_mutex_lock( &mi->mi_cache_lock ); mc = ( monitor_cache_t * )ldap_avl_find( mi->mi_cache, ( caddr_t )&tmp_mc, monitor_cache_cmp ); if ( mc != NULL ) { /* entry is returned with mutex locked */ monitor_cache_lock( mc->mc_e ); *ep = mc->mc_e; } ldap_pvt_thread_mutex_unlock( &mi->mi_cache_lock ); return ( *ep == NULL ? -1 : 0 ); } /* * gets an entry from the cache based on the normalized dn * with mutex locked */ int monitor_cache_remove( monitor_info_t *mi, struct berval *ndn, Entry **ep ) { monitor_cache_t tmp_mc, *mc; struct berval pndn; assert( mi != NULL ); assert( ndn != NULL ); assert( ep != NULL ); *ep = NULL; dnParent( ndn, &pndn ); retry:; ldap_pvt_thread_mutex_lock( &mi->mi_cache_lock ); tmp_mc.mc_ndn = *ndn; mc = ( monitor_cache_t * )ldap_avl_find( mi->mi_cache, ( caddr_t )&tmp_mc, monitor_cache_cmp ); if ( mc != NULL ) { monitor_cache_t *pmc; tmp_mc.mc_ndn = pndn; pmc = ( monitor_cache_t * )ldap_avl_find( mi->mi_cache, ( caddr_t )&tmp_mc, monitor_cache_cmp ); if ( pmc != NULL ) { monitor_entry_t *mp = (monitor_entry_t *)mc->mc_e->e_private, *pmp = (monitor_entry_t *)pmc->mc_e->e_private; Entry **entryp, *prev = NULL; monitor_cache_lock( pmc->mc_e ); for ( entryp = &pmp->mp_children; *entryp != NULL; ) { monitor_entry_t *next = (monitor_entry_t *)(*entryp)->e_private; monitor_cache_lock( *entryp ); if ( next == mp ) { if ( mc->mc_e == pmp->mp_last ) { pmp->mp_last = prev; } *entryp = next->mp_next; entryp = NULL; break; } if ( prev != NULL ) { monitor_cache_release( mi, prev ); } prev = *entryp; entryp = &next->mp_next; } if ( prev ) { monitor_cache_release( mi, prev ); } if ( entryp != NULL ) { Debug( LDAP_DEBUG_ANY, "monitor_cache_remove(\"%s\"): " "not in parent's list\n", ndn->bv_val ); } /* either succeeded, and the entry is no longer * in its parent's list, or failed, and the * entry is neither mucked with nor returned */ monitor_cache_release( mi, pmc->mc_e ); if ( entryp == NULL ) { monitor_cache_t *tmpmc; tmp_mc.mc_ndn = *ndn; tmpmc = ldap_avl_delete( &mi->mi_cache, ( caddr_t )&tmp_mc, monitor_cache_cmp ); assert( tmpmc == mc ); *ep = mc->mc_e; ch_free( mc ); mc = NULL; /* NOTE: we destroy the mutex, but otherwise * leave the private data around; specifically, * callbacks need be freed by someone else */ ldap_pvt_thread_mutex_destroy( &mp->mp_mutex ); mp->mp_next = NULL; mp->mp_children = NULL; mp->mp_last = NULL; } } if ( mc ) { monitor_cache_release( mi, mc->mc_e ); } } ldap_pvt_thread_mutex_unlock( &mi->mi_cache_lock ); return ( *ep == NULL ? -1 : 0 ); } /* * If the entry exists in cache, it is returned in locked status; * otherwise, if the parent exists, if it may generate volatile * descendants an attempt to generate the required entry is * performed and, if successful, the entry is returned */ int monitor_cache_dn2entry( Operation *op, SlapReply *rs, struct berval *ndn, Entry **ep, Entry **matched ) { monitor_info_t *mi = (monitor_info_t *)op->o_bd->be_private; int rc; struct berval p_ndn = BER_BVNULL; Entry *e_parent; monitor_entry_t *mp; assert( mi != NULL ); assert( ndn != NULL ); assert( ep != NULL ); assert( matched != NULL ); *matched = NULL; if ( !dnIsSuffix( ndn, &op->o_bd->be_nsuffix[ 0 ] ) ) { return( -1 ); } rc = monitor_cache_get( mi, ndn, ep ); if ( !rc && *ep != NULL ) { return( 0 ); } /* try with parent/ancestors */ if ( BER_BVISNULL( ndn ) ) { BER_BVSTR( &p_ndn, "" ); } else { dnParent( ndn, &p_ndn ); } rc = monitor_cache_dn2entry( op, rs, &p_ndn, &e_parent, matched ); if ( rc || e_parent == NULL ) { return( -1 ); } mp = ( monitor_entry_t * )e_parent->e_private; rc = -1; if ( mp->mp_flags & MONITOR_F_VOLATILE_CH ) { /* parent entry generates volatile children */ rc = monitor_entry_create( op, rs, ndn, e_parent, ep ); } if ( !rc ) { monitor_cache_lock( *ep ); monitor_cache_release( mi, e_parent ); } else { *matched = e_parent; } return( rc ); } /* * releases the lock of the entry; if it is marked as volatile, it is * destroyed. */ int monitor_cache_release( monitor_info_t *mi, Entry *e ) { monitor_entry_t *mp; assert( mi != NULL ); assert( e != NULL ); assert( e->e_private != NULL ); mp = ( monitor_entry_t * )e->e_private; if ( mp->mp_flags & MONITOR_F_VOLATILE ) { ldap_pvt_thread_mutex_unlock( &mp->mp_mutex ); ldap_pvt_thread_mutex_destroy( &mp->mp_mutex ); ch_free( mp ); e->e_private = NULL; entry_free( e ); return( 0 ); } ldap_pvt_thread_mutex_unlock( &mp->mp_mutex ); return( 0 ); } static void monitor_entry_destroy( void *v_mc ) { monitor_cache_t *mc = (monitor_cache_t *)v_mc; if ( mc->mc_e != NULL ) { monitor_entry_t *mp; assert( mc->mc_e->e_private != NULL ); mp = ( monitor_entry_t * )mc->mc_e->e_private; if ( mp->mp_cb ) { monitor_callback_t *cb; for ( cb = mp->mp_cb; cb != NULL; ) { monitor_callback_t *next = cb->mc_next; if ( cb->mc_free ) { (void)cb->mc_free( mc->mc_e, &cb->mc_private ); } ch_free( mp->mp_cb ); cb = next; } } ldap_pvt_thread_mutex_destroy( &mp->mp_mutex ); ch_free( mp ); mc->mc_e->e_private = NULL; entry_free( mc->mc_e ); } ch_free( mc ); } int monitor_cache_destroy( monitor_info_t *mi ) { if ( mi->mi_cache ) { ldap_avl_free( mi->mi_cache, monitor_entry_destroy ); } return 0; } int monitor_back_release( Operation *op, Entry *e, int rw ) { monitor_info_t *mi = ( monitor_info_t * )op->o_bd->be_private; return monitor_cache_release( mi, e ); }