Diff for /servers/slapd/overlays/syncprov.c between versions 1.147.2.79 and 1.293

version 1.147.2.79, 2011/01/03 22:37:33 version 1.293, 2009/11/11 03:03:27
Line 1 Line 1
 /* $OpenLDAP$ */  /* $OpenLDAP: pkg/ldap/servers/slapd/overlays/syncprov.c,v 1.292 2009/11/11 02:36:46 hyc Exp $ */
 /* syncprov.c - syncrepl provider */  /* syncprov.c - syncrepl provider */
 /* This work is part of OpenLDAP Software <http://www.openldap.org/>.  /* This work is part of OpenLDAP Software <http://www.openldap.org/>.
  *   *
  * Copyright 2004-2010 The OpenLDAP Foundation.   * Copyright 2004-2009 The OpenLDAP Foundation.
  * All rights reserved.   * All rights reserved.
  *   *
  * Redistribution and use in source and binary forms, with or without   * Redistribution and use in source and binary forms, with or without
Line 48  typedef struct modtarget { Line 48  typedef struct modtarget {
 /* A queued result of a persistent search */  /* A queued result of a persistent search */
 typedef struct syncres {  typedef struct syncres {
         struct syncres *s_next;          struct syncres *s_next;
         Entry *s_e;  
         struct berval s_dn;          struct berval s_dn;
         struct berval s_ndn;          struct berval s_ndn;
         struct berval s_uuid;          struct berval s_uuid;
Line 133  typedef struct syncprov_info_t { Line 132  typedef struct syncprov_info_t {
         int             si_numops;      /* number of ops since last checkpoint */          int             si_numops;      /* number of ops since last checkpoint */
         int             si_nopres;      /* Skip present phase */          int             si_nopres;      /* Skip present phase */
         int             si_usehint;     /* use reload hint */          int             si_usehint;     /* use reload hint */
         int             si_active;      /* True if there are active mods */  
         int             si_dirty;       /* True if the context is dirty, i.e changes  
                                                  * have been made without updating the csn. */  
         time_t  si_chklast;     /* time of last checkpoint */          time_t  si_chklast;     /* time of last checkpoint */
         Avlnode *si_mods;       /* entries being modified */          Avlnode *si_mods;       /* entries being modified */
         sessionlog      *si_logs;          sessionlog      *si_logs;
Line 149  typedef struct opcookie { Line 145  typedef struct opcookie {
         slap_overinst *son;          slap_overinst *son;
         syncmatches *smatches;          syncmatches *smatches;
         modtarget *smt;          modtarget *smt;
         Entry *se;  
         struct berval sdn;      /* DN of entry, for deletes */          struct berval sdn;      /* DN of entry, for deletes */
         struct berval sndn;          struct berval sndn;
         struct berval suuid;    /* UUID of entry */          struct berval suuid;    /* UUID of entry */
Line 159  typedef struct opcookie { Line 154  typedef struct opcookie {
         short sreference;       /* Is the entry a reference? */          short sreference;       /* Is the entry a reference? */
 } opcookie;  } opcookie;
   
 typedef struct mutexint {  
         ldap_pvt_thread_mutex_t mi_mutex;  
         int mi_int;  
 } mutexint;  
   
 typedef struct fbase_cookie {  typedef struct fbase_cookie {
         struct berval *fdn;     /* DN of a modified entry, for scope testing */          struct berval *fdn;     /* DN of a modified entry, for scope testing */
         syncops *fss;   /* persistent search we're testing against */          syncops *fss;   /* persistent search we're testing against */
Line 749  again: Line 739  again:
         return rc;          return rc;
 }  }
   
 /* Should find a place to cache these */  
 static mutexint *get_mutexint()  
 {  
         mutexint *mi = ch_malloc( sizeof( mutexint ));  
         ldap_pvt_thread_mutex_init( &mi->mi_mutex );  
         mi->mi_int = 1;  
         return mi;  
 }  
   
 static void inc_mutexint( mutexint *mi )  
 {  
         ldap_pvt_thread_mutex_lock( &mi->mi_mutex );  
         mi->mi_int++;  
         ldap_pvt_thread_mutex_unlock( &mi->mi_mutex );  
 }  
   
 /* return resulting counter */  
 static int dec_mutexint( mutexint *mi )  
 {  
         int i;  
         ldap_pvt_thread_mutex_lock( &mi->mi_mutex );  
         i = --mi->mi_int;  
         ldap_pvt_thread_mutex_unlock( &mi->mi_mutex );  
         if ( !i ) {  
                 ldap_pvt_thread_mutex_destroy( &mi->mi_mutex );  
                 ch_free( mi );  
         }  
         return i;  
 }  
   
 static void  static void
 syncprov_free_syncop( syncops *so )  syncprov_free_syncop( syncops *so )
 {  {
Line 802  syncprov_free_syncop( syncops *so ) Line 762  syncprov_free_syncop( syncops *so )
         ch_free( so->s_base.bv_val );          ch_free( so->s_base.bv_val );
         for ( sr=so->s_res; sr; sr=srnext ) {          for ( sr=so->s_res; sr; sr=srnext ) {
                 srnext = sr->s_next;                  srnext = sr->s_next;
                 if ( sr->s_e ) {  
                         if ( !dec_mutexint( sr->s_e->e_private )) {  
                                 sr->s_e->e_private = NULL;  
                                 entry_free( sr->s_e );  
                         }  
                 }  
                 ch_free( sr );                  ch_free( sr );
         }          }
         ldap_pvt_thread_mutex_destroy( &so->s_mutex );          ldap_pvt_thread_mutex_destroy( &so->s_mutex );
Line 816  syncprov_free_syncop( syncops *so ) Line 770  syncprov_free_syncop( syncops *so )
   
 /* Send a persistent search response */  /* Send a persistent search response */
 static int  static int
 syncprov_sendresp( Operation *op, opcookie *opc, syncops *so, int mode )  syncprov_sendresp( Operation *op, opcookie *opc, syncops *so,
           Entry **e, int mode )
 {  {
         slap_overinst *on = opc->son;          slap_overinst *on = opc->son;
   
Line 837  syncprov_sendresp( Operation *op, opcook Line 792  syncprov_sendresp( Operation *op, opcook
         }          }
   
 #ifdef LDAP_DEBUG  #ifdef LDAP_DEBUG
         if ( so->s_sid > 0 ) {          if ( !BER_BVISNULL( &cookie )) {
                 Debug( LDAP_DEBUG_SYNC, "syncprov_sendresp: to=%03x, cookie=%s\n",                  if ( so->s_sid > 0 ) {
                         so->s_sid, cookie.bv_val ? cookie.bv_val : "", 0 );                          Debug( LDAP_DEBUG_SYNC, "syncprov_sendresp: to=%03x, cookie=%s\n",
         } else {                                  so->s_sid, cookie.bv_val , 0 );
                 Debug( LDAP_DEBUG_SYNC, "syncprov_sendresp: cookie=%s\n",                  } else {
                         cookie.bv_val ? cookie.bv_val : "", 0, 0 );                          Debug( LDAP_DEBUG_SYNC, "syncprov_sendresp: cookie=%s\n",
         }                                  cookie.bv_val, 0, 0 );
                   }
           }               
 #endif  #endif
   
         e_uuid.e_attrs = &a_uuid;          e_uuid.e_attrs = &a_uuid;
Line 856  syncprov_sendresp( Operation *op, opcook Line 813  syncprov_sendresp( Operation *op, opcook
         }          }
   
         rs.sr_ctrls = ctrls;          rs.sr_ctrls = ctrls;
         rs.sr_entry = &e_uuid;          op->o_bd->bd_info = (BackendInfo *)on->on_info;
         if ( mode == LDAP_SYNC_ADD || mode == LDAP_SYNC_MODIFY ) {  
                 e_uuid = *opc->se;  
                 e_uuid.e_private = NULL;  
         }  
   
         switch( mode ) {          switch( mode ) {
         case LDAP_SYNC_ADD:          case LDAP_SYNC_ADD:
                   rs.sr_entry = *e;
                   if ( rs.sr_entry->e_private )
                           rs.sr_flags = REP_ENTRY_MUSTRELEASE;
                 if ( opc->sreference && so->s_op->o_managedsait <= SLAP_CONTROL_IGNORED ) {                  if ( opc->sreference && so->s_op->o_managedsait <= SLAP_CONTROL_IGNORED ) {
                         rs.sr_ref = get_entry_referrals( op, rs.sr_entry );                          rs.sr_ref = get_entry_referrals( op, rs.sr_entry );
                         rs.sr_err = send_search_reference( op, &rs );                          rs.sr_err = send_search_reference( op, &rs );
                         ber_bvarray_free( rs.sr_ref );                          ber_bvarray_free( rs.sr_ref );
                           if ( !rs.sr_entry )
                                   *e = NULL;
                         break;                          break;
                 }                  }
                 /* fallthru */                  /* fallthru */
         case LDAP_SYNC_MODIFY:          case LDAP_SYNC_MODIFY:
                   rs.sr_entry = *e;
                   if ( rs.sr_entry->e_private )
                           rs.sr_flags = REP_ENTRY_MUSTRELEASE;
                 rs.sr_attrs = op->ors_attrs;                  rs.sr_attrs = op->ors_attrs;
                 rs.sr_err = send_search_entry( op, &rs );                  rs.sr_err = send_search_entry( op, &rs );
                   if ( !rs.sr_entry )
                           *e = NULL;
                 break;                  break;
         case LDAP_SYNC_DELETE:          case LDAP_SYNC_DELETE:
                 e_uuid.e_attrs = NULL;                  e_uuid.e_attrs = NULL;
                 e_uuid.e_name = opc->sdn;                  e_uuid.e_name = opc->sdn;
                 e_uuid.e_nname = opc->sndn;                  e_uuid.e_nname = opc->sndn;
                   rs.sr_entry = &e_uuid;
                 if ( opc->sreference && so->s_op->o_managedsait <= SLAP_CONTROL_IGNORED ) {                  if ( opc->sreference && so->s_op->o_managedsait <= SLAP_CONTROL_IGNORED ) {
                         struct berval bv = BER_BVNULL;                          struct berval bv = BER_BVNULL;
                         rs.sr_ref = &bv;                          rs.sr_ref = &bv;
Line 892  syncprov_sendresp( Operation *op, opcook Line 855  syncprov_sendresp( Operation *op, opcook
         }          }
         /* In case someone else freed it already? */          /* In case someone else freed it already? */
         if ( rs.sr_ctrls ) {          if ( rs.sr_ctrls ) {
                 int i;                  op->o_tmpfree( rs.sr_ctrls[0], op->o_tmpmemctx );
                 for ( i=0; rs.sr_ctrls[i]; i++ ) {  
                         if ( rs.sr_ctrls[i] == ctrls[0] ) {  
                                 op->o_tmpfree( ctrls[0]->ldctl_value.bv_val, op->o_tmpmemctx );  
                                 ctrls[0]->ldctl_value.bv_val = NULL;  
                                 break;  
                         }  
                 }  
                 rs.sr_ctrls = NULL;                  rs.sr_ctrls = NULL;
         }          }
   
Line 934  syncprov_qplay( Operation *op, syncops * Line 890  syncprov_qplay( Operation *op, syncops *
                 ldap_pvt_thread_mutex_unlock( &so->s_mutex );                  ldap_pvt_thread_mutex_unlock( &so->s_mutex );
   
                 if ( sr->s_mode == LDAP_SYNC_NEW_COOKIE ) {                  if ( sr->s_mode == LDAP_SYNC_NEW_COOKIE ) {
                         SlapReply rs = { REP_INTERMEDIATE };                      SlapReply rs = { REP_INTERMEDIATE };
   
                         rc = syncprov_sendinfo( op, &rs, LDAP_TAG_SYNC_NEW_COOKIE,                      rc = syncprov_sendinfo( op, &rs, LDAP_TAG_SYNC_NEW_COOKIE,
                                 &sr->s_csn, 0, NULL, 0 );                                  &sr->s_csn, 0, NULL, 0 );
                 } else {                  } else {
                         opc.sdn = sr->s_dn;                          opc.sdn = sr->s_dn;
Line 944  syncprov_qplay( Operation *op, syncops * Line 900  syncprov_qplay( Operation *op, syncops *
                         opc.suuid = sr->s_uuid;                          opc.suuid = sr->s_uuid;
                         opc.sctxcsn = sr->s_csn;                          opc.sctxcsn = sr->s_csn;
                         opc.sreference = sr->s_isreference;                          opc.sreference = sr->s_isreference;
                         opc.se = sr->s_e;                          e = NULL;
   
                         rc = syncprov_sendresp( op, &opc, so, sr->s_mode );                          if ( sr->s_mode != LDAP_SYNC_DELETE ) {
                                   rc = overlay_entry_get_ov( op, &opc.sndn, NULL, NULL, 0, &e, on );
                                   if ( rc ) {
                                           Debug( LDAP_DEBUG_SYNC, "syncprov_qplay: failed to get %s, "
                                                   "error (%d), ignoring...\n", opc.sndn.bv_val, rc, 0 );
                                           ch_free( sr );
                                           rc = 0;
                                           continue;
                                   }
                           }
                           rc = syncprov_sendresp( op, &opc, so, &e, sr->s_mode );
   
                 }                          if ( e ) {
                 if ( sr->s_e ) {                                  overlay_entry_release_ov( op, e, 0, on );
                         if ( !dec_mutexint( sr->s_e->e_private )) {  
                                 sr->s_e->e_private = NULL;  
                                 entry_free ( sr->s_e );  
                         }                          }
                 }                  }
   
Line 1046  syncprov_qresp( opcookie *opc, syncops * Line 1009  syncprov_qresp( opcookie *opc, syncops *
                 srsize += cookie.bv_len + 1;                  srsize += cookie.bv_len + 1;
         sr = ch_malloc( srsize );          sr = ch_malloc( srsize );
         sr->s_next = NULL;          sr->s_next = NULL;
         sr->s_e = opc->se;  
         /* bump refcount on this entry */  
         if ( opc->se )  
                 inc_mutexint( opc->se->e_private );  
         sr->s_dn.bv_val = (char *)(sr + 1);          sr->s_dn.bv_val = (char *)(sr + 1);
         sr->s_dn.bv_len = opc->sdn.bv_len;          sr->s_dn.bv_len = opc->sdn.bv_len;
         sr->s_mode = mode;          sr->s_mode = mode;
Line 1193  syncprov_matchops( Operation *op, opcook Line 1152  syncprov_matchops( Operation *op, opcook
                 rc = overlay_entry_get_ov( op, fc.fdn, NULL, NULL, 0, &e, on );                  rc = overlay_entry_get_ov( op, fc.fdn, NULL, NULL, 0, &e, on );
                 /* If we're sending responses now, make a copy and unlock the DB */                  /* If we're sending responses now, make a copy and unlock the DB */
                 if ( e && !saveit ) {                  if ( e && !saveit ) {
                         if ( !opc->se ) {                          Entry *e2 = entry_dup( e );
                                 opc->se = entry_dup( e );  
                                 opc->se->e_private = get_mutexint();  
                         }  
                         overlay_entry_release_ov( op, e, 0, on );                          overlay_entry_release_ov( op, e, 0, on );
                         e = opc->se;                          e = e2;
                 }                  }
                 if ( rc ) {                  if ( rc ) {
                         op->o_bd = b0;                          op->o_bd = b0;
Line 1206  syncprov_matchops( Operation *op, opcook Line 1162  syncprov_matchops( Operation *op, opcook
                 }                  }
         } else {          } else {
                 e = op->ora_e;                  e = op->ora_e;
                 if ( !saveit ) {  
                         if ( !opc->se ) {  
                                 opc->se = entry_dup( e );  
                                 opc->se->e_private = get_mutexint();  
                         }  
                         e = opc->se;  
                 }  
         }          }
   
         if ( saveit || op->o_tag == LDAP_REQ_ADD ) {          if ( saveit || op->o_tag == LDAP_REQ_ADD ) {
Line 1242  syncprov_matchops( Operation *op, opcook Line 1191  syncprov_matchops( Operation *op, opcook
                 if ( ss->s_op->o_abandon )                  if ( ss->s_op->o_abandon )
                         continue;                          continue;
   
                 /* Don't send ops back to the originator */                  /* First time thru, check for possible skips */
                 if ( opc->osid > 0 && opc->osid == ss->s_sid ) {                  if ( saveit || op->o_tag == LDAP_REQ_ADD ) {
                         Debug( LDAP_DEBUG_SYNC, "syncprov_matchops: skipping original sid %03x\n",  
                                 opc->osid, 0, 0 );  
                         continue;  
                 }  
   
                 /* Don't send ops back to the messenger */                          /* Don't send ops back to the originator */
                 if ( opc->rsid > 0 && opc->rsid == ss->s_sid ) {                          if ( opc->osid > 0 && opc->osid == ss->s_sid ) {
                         Debug( LDAP_DEBUG_SYNC, "syncprov_matchops: skipping relayed sid %03x\n",                                  Debug( LDAP_DEBUG_SYNC, "syncprov_matchops: skipping original sid %03x\n",
                                 opc->rsid, 0, 0 );                                          opc->osid, 0, 0 );
                         continue;                                  continue;
                           }
   
                           /* Don't send ops back to the messenger */
                           if ( opc->rsid > 0 && opc->rsid == ss->s_sid ) {
                                   Debug( LDAP_DEBUG_SYNC, "syncprov_matchops: skipping relayed sid %03x\n",
                                           opc->rsid, 0, 0 );
                                   continue;
                           }
                 }                  }
   
                 /* validate base */                  /* validate base */
Line 1273  syncprov_matchops( Operation *op, opcook Line 1226  syncprov_matchops( Operation *op, opcook
                         continue;                          continue;
                 }                  }
   
   
                 /* If we're sending results now, look for this op in old matches */                  /* If we're sending results now, look for this op in old matches */
                 if ( !saveit ) {                  if ( !saveit ) {
                         syncmatches *old;                          syncmatches *old;
Line 1296  syncprov_matchops( Operation *op, opcook Line 1250  syncprov_matchops( Operation *op, opcook
                 }                  }
   
                 if ( fc.fscope ) {                  if ( fc.fscope ) {
                         ldap_pvt_thread_mutex_lock( &ss->s_mutex );  
                         op2 = *ss->s_op;                          op2 = *ss->s_op;
                         oh = *op->o_hdr;                          oh = *op->o_hdr;
                         oh.oh_conn = ss->s_op->o_conn;                          oh.oh_conn = ss->s_op->o_conn;
Line 1305  syncprov_matchops( Operation *op, opcook Line 1258  syncprov_matchops( Operation *op, opcook
                         op2.o_hdr = &oh;                          op2.o_hdr = &oh;
                         op2.o_extra = op->o_extra;                          op2.o_extra = op->o_extra;
                         op2.o_callback = NULL;                          op2.o_callback = NULL;
                         if (ss->s_flags & PS_FIX_FILTER) {                          rc = test_filter( &op2, e, ss->s_op->ors_filter );
                                 /* Skip the AND/GE clause that we stuck on in front. We  
                                    would lose deletes/mods that happen during the refresh  
                                    phase otherwise (ITS#6555) */  
                                 op2.ors_filter = ss->s_op->ors_filter->f_and->f_next;  
                         }  
                         ldap_pvt_thread_mutex_unlock( &ss->s_mutex );  
                         rc = test_filter( &op2, e, op2.ors_filter );  
                 }                  }
   
                 Debug( LDAP_DEBUG_TRACE, "syncprov_matchops: sid %03x fscope %d rc %d\n",                  Debug( LDAP_DEBUG_TRACE, "syncprov_matchops: sid %03x fscope %d rc %d\n",
Line 1352  syncprov_matchops( Operation *op, opcook Line 1298  syncprov_matchops( Operation *op, opcook
                 if ( !SLAP_ISOVERLAY( op->o_bd )) {                  if ( !SLAP_ISOVERLAY( op->o_bd )) {
                         op->o_bd = &db;                          op->o_bd = &db;
                 }                  }
                 if ( saveit )                  overlay_entry_release_ov( op, e, 0, on );
                         overlay_entry_release_ov( op, e, 0, on );  
                 op->o_bd = b0;                  op->o_bd = b0;
         }          }
         if ( opc->se && !saveit ) {  
                 if ( !dec_mutexint( opc->se->e_private )) {  
                         opc->se->e_private = NULL;  
                         entry_free( opc->se );  
                         opc->se = NULL;  
                 }  
         }  
         if ( freefdn ) {          if ( freefdn ) {
                 op->o_tmpfree( fc.fdn->bv_val, op->o_tmpmemctx );                  op->o_tmpfree( fc.fdn->bv_val, op->o_tmpmemctx );
         }          }
Line 1379  syncprov_op_cleanup( Operation *op, Slap Line 1317  syncprov_op_cleanup( Operation *op, Slap
         syncmatches *sm, *snext;          syncmatches *sm, *snext;
         modtarget *mt, mtdummy;          modtarget *mt, mtdummy;
   
         ldap_pvt_thread_mutex_lock( &si->si_ops_mutex );  
         if ( si->si_active )  
                 si->si_active--;  
         ldap_pvt_thread_mutex_unlock( &si->si_ops_mutex );  
   
         for (sm = opc->smatches; sm; sm=snext) {          for (sm = opc->smatches; sm; sm=snext) {
                 snext = sm->sm_next;                  snext = sm->sm_next;
                 syncprov_free_syncop( sm->sm_op );                  syncprov_free_syncop( sm->sm_op );
Line 1429  syncprov_checkpoint( Operation *op, Slap Line 1362  syncprov_checkpoint( Operation *op, Slap
         SlapReply rsm = { 0 };          SlapReply rsm = { 0 };
         slap_callback cb = {0};          slap_callback cb = {0};
         BackendDB be;          BackendDB be;
         BackendInfo *bi;          struct berval ctxdn;
   
 #ifdef CHECK_CSN  #ifdef CHECK_CSN
         Syntax *syn = slap_schema.si_ad_contextCSN->ad_type->sat_syntax;          Syntax *syn = slap_schema.si_ad_contextCSN->ad_type->sat_syntax;
Line 1456  syncprov_checkpoint( Operation *op, Slap Line 1389  syncprov_checkpoint( Operation *op, Slap
         if ( SLAP_GLUE_SUBORDINATE( op->o_bd )) {          if ( SLAP_GLUE_SUBORDINATE( op->o_bd )) {
                 be = *on->on_info->oi_origdb;                  be = *on->on_info->oi_origdb;
                 opm.o_bd = &be;                  opm.o_bd = &be;
                   ctxdn = be.be_nsuffix[0];
           } else {
                   ctxdn = si->si_contextdn;
         }          }
         opm.o_req_dn = si->si_contextdn;          opm.o_req_dn = ctxdn;
         opm.o_req_ndn = si->si_contextdn;          opm.o_req_ndn = ctxdn;
         bi = opm.o_bd->bd_info;  
         opm.o_bd->bd_info = on->on_info->oi_orig;          opm.o_bd->bd_info = on->on_info->oi_orig;
         opm.o_managedsait = SLAP_CONTROL_NONCRITICAL;          opm.o_managedsait = SLAP_CONTROL_NONCRITICAL;
         opm.o_no_schema_check = 1;          opm.o_no_schema_check = 1;
Line 1477  syncprov_checkpoint( Operation *op, Slap Line 1412  syncprov_checkpoint( Operation *op, Slap
                 if ( e == opm.ora_e )                  if ( e == opm.ora_e )
                         be_entry_release_w( &opm, opm.ora_e );                          be_entry_release_w( &opm, opm.ora_e );
         }          }
         opm.o_bd->bd_info = bi;  
   
         if ( mod.sml_next != NULL ) {          if ( mod.sml_next != NULL ) {
                 slap_mods_free( mod.sml_next, 1 );                  slap_mods_free( mod.sml_next, 1 );
Line 1740  syncprov_op_response( Operation *op, Sla Line 1674  syncprov_op_response( Operation *op, Sla
                 maxcsn.bv_len = sizeof(cbuf);                  maxcsn.bv_len = sizeof(cbuf);
                 ldap_pvt_thread_rdwr_wlock( &si->si_csn_rwlock );                  ldap_pvt_thread_rdwr_wlock( &si->si_csn_rwlock );
   
                 slap_get_commit_csn( op, &maxcsn, &foundit );                  if ( op->o_dont_replicate && op->o_tag == LDAP_REQ_MODIFY &&
                 if ( BER_BVISEMPTY( &maxcsn ) && SLAP_GLUE_SUBORDINATE( op->o_bd )) {  
                         /* syncrepl queues the CSN values in the db where  
                          * it is configured , not where the changes are made.  
                          * So look for a value in the glue db if we didn't  
                          * find any in this db.  
                          */  
                         BackendDB *be = op->o_bd;  
                         op->o_bd = select_backend( &be->be_nsuffix[0], 1);  
                         maxcsn.bv_val = cbuf;  
                         maxcsn.bv_len = sizeof(cbuf);  
                         slap_get_commit_csn( op, &maxcsn, &foundit );  
                         op->o_bd = be;  
                 }  
                 if ( !BER_BVISEMPTY( &maxcsn ) ) {  
                         int i, sid;  
 #ifdef CHECK_CSN  
                         Syntax *syn = slap_schema.si_ad_contextCSN->ad_type->sat_syntax;  
                         assert( !syn->ssyn_validate( syn, &maxcsn ));  
 #endif  
                         sid = slap_parse_csn_sid( &maxcsn );  
                         for ( i=0; i<si->si_numcsns; i++ ) {  
                                 if ( sid == si->si_sids[i] ) {  
                                         if ( ber_bvcmp( &maxcsn, &si->si_ctxcsn[i] ) > 0 ) {  
                                                 ber_bvreplace( &si->si_ctxcsn[i], &maxcsn );  
                                                 csn_changed = 1;  
                                         }  
                                         break;  
                                 }  
                         }  
                         /* It's a new SID for us */  
                         if ( i == si->si_numcsns ) {  
                                 value_add_one( &si->si_ctxcsn, &maxcsn );  
                                 csn_changed = 1;  
                                 si->si_numcsns++;  
                                 si->si_sids = ch_realloc( si->si_sids, si->si_numcsns *  
                                         sizeof(int));  
                                 si->si_sids[i] = sid;  
                         }  
                 }  
   
                 /* Don't do any processing for consumer contextCSN updates */  
                 if ( op->o_dont_replicate ) {  
                         if ( op->o_tag == LDAP_REQ_MODIFY &&  
                                 op->orm_modlist->sml_op == LDAP_MOD_REPLACE &&                                  op->orm_modlist->sml_op == LDAP_MOD_REPLACE &&
                                 op->orm_modlist->sml_desc == slap_schema.si_ad_contextCSN ) {                                  op->orm_modlist->sml_desc == slap_schema.si_ad_contextCSN ) {
                         /* Catch contextCSN updates from syncrepl. We have to look at                          /* Catch contextCSN updates from syncrepl. We have to look at
Line 1795  syncprov_op_response( Operation *op, Sla Line 1686  syncprov_op_response( Operation *op, Sla
   
                         for ( i=0; i<mod->sml_numvals; i++ ) {                          for ( i=0; i<mod->sml_numvals; i++ ) {
                                 sid = slap_parse_csn_sid( &mod->sml_values[i] );                                  sid = slap_parse_csn_sid( &mod->sml_values[i] );
   
                                 for ( j=0; j<si->si_numcsns; j++ ) {                                  for ( j=0; j<si->si_numcsns; j++ ) {
                                         if ( sid == si->si_sids[j] ) {                                          if ( sid == si->si_sids[j] ) {
                                                 if ( ber_bvcmp( &mod->sml_values[i], &si->si_ctxcsn[j] ) > 0 ) {                                                  if ( ber_bvcmp( &mod->sml_values[i], &si->si_ctxcsn[j] ) > 0 ) {
Line 1814  syncprov_op_response( Operation *op, Sla Line 1706  syncprov_op_response( Operation *op, Sla
                                         csn_changed = 1;                                          csn_changed = 1;
                                 }                                  }
                         }                          }
                         if ( csn_changed )  
                                 si->si_dirty = 0;  
                         ldap_pvt_thread_rdwr_wunlock( &si->si_csn_rwlock );                          ldap_pvt_thread_rdwr_wunlock( &si->si_csn_rwlock );
   
                         if ( csn_changed ) {                          if ( csn_changed ) {
Line 1831  syncprov_op_response( Operation *op, Sla Line 1721  syncprov_op_response( Operation *op, Sla
                                         }                                          }
                                 }                                  }
                         }                          }
                         } else {                          goto leave;
                         ldap_pvt_thread_rdwr_wunlock( &si->si_csn_rwlock );                  }
   
                   slap_get_commit_csn( op, &maxcsn, &foundit );
                   if ( BER_BVISEMPTY( &maxcsn ) && SLAP_GLUE_SUBORDINATE( op->o_bd )) {
                           /* syncrepl queues the CSN values in the db where
                            * it is configured , not where the changes are made.
                            * So look for a value in the glue db if we didn't
                            * find any in this db.
                            */
                           BackendDB *be = op->o_bd;
                           op->o_bd = select_backend( &be->be_nsuffix[0], 1);
                           maxcsn.bv_val = cbuf;
                           maxcsn.bv_len = sizeof(cbuf);
                           slap_get_commit_csn( op, &maxcsn, &foundit );
                           op->o_bd = be;
                   }
                   if ( !BER_BVISEMPTY( &maxcsn ) ) {
                           int i, sid;
   #ifdef CHECK_CSN
                           Syntax *syn = slap_schema.si_ad_contextCSN->ad_type->sat_syntax;
                           assert( !syn->ssyn_validate( syn, &maxcsn ));
   #endif
                           sid = slap_parse_csn_sid( &maxcsn );
                           for ( i=0; i<si->si_numcsns; i++ ) {
                                   if ( sid == si->si_sids[i] ) {
                                           if ( ber_bvcmp( &maxcsn, &si->si_ctxcsn[i] ) > 0 ) {
                                                   ber_bvreplace( &si->si_ctxcsn[i], &maxcsn );
                                                   csn_changed = 1;
                                           }
                                           break;
                                   }
                         }                          }
                           /* It's a new SID for us */
                           if ( i == si->si_numcsns ) {
                                   value_add_one( &si->si_ctxcsn, &maxcsn );
                                   csn_changed = 1;
                                   si->si_numcsns++;
                                   si->si_sids = ch_realloc( si->si_sids, si->si_numcsns *
                                           sizeof(int));
                                   si->si_sids[i] = sid;
                           }
   #if 0
                   } else if ( !foundit ) {
                           /* internal ops that aren't meant to be replicated */
                           ldap_pvt_thread_rdwr_wunlock( &si->si_csn_rwlock );
                           return SLAP_CB_CONTINUE;
   #endif
                   }
   
                   /* Don't do any processing for consumer contextCSN updates */
                   if ( op->o_dont_replicate ) {
                           ldap_pvt_thread_rdwr_wunlock( &si->si_csn_rwlock );
                         goto leave;                          goto leave;
                 }                  }
   
Line 1859  syncprov_op_response( Operation *op, Sla Line 1799  syncprov_op_response( Operation *op, Sla
                                 }                                  }
                         }                          }
                 }                  }
                 si->si_dirty = !csn_changed;  
                 ldap_pvt_thread_rdwr_wunlock( &si->si_csn_rwlock );                  ldap_pvt_thread_rdwr_wunlock( &si->si_csn_rwlock );
   
                 if ( do_check ) {                  if ( do_check ) {
Line 1987  syncprov_op_mod( Operation *op, SlapRepl Line 1926  syncprov_op_mod( Operation *op, SlapRepl
   
         ldap_pvt_thread_mutex_lock( &si->si_ops_mutex );          ldap_pvt_thread_mutex_lock( &si->si_ops_mutex );
         have_psearches = ( si->si_ops != NULL );          have_psearches = ( si->si_ops != NULL );
         si->si_active++;  
         ldap_pvt_thread_mutex_unlock( &si->si_ops_mutex );          ldap_pvt_thread_mutex_unlock( &si->si_ops_mutex );
   
         cbsize = sizeof(slap_callback) + sizeof(opcookie) +          cbsize = sizeof(slap_callback) + sizeof(opcookie) +
Line 2378  syncprov_op_search( Operation *op, SlapR Line 2316  syncprov_op_search( Operation *op, SlapR
         BerVarray ctxcsn;          BerVarray ctxcsn;
         int i, *sids, numcsns;          int i, *sids, numcsns;
         struct berval mincsn;          struct berval mincsn;
         int dirty = 0;  
   
         if ( !(op->o_sync_mode & SLAP_SYNC_REFRESH) ) return SLAP_CB_CONTINUE;          if ( !(op->o_sync_mode & SLAP_SYNC_REFRESH) ) return SLAP_CB_CONTINUE;
   
Line 2423  syncprov_op_search( Operation *op, SlapR Line 2360  syncprov_op_search( Operation *op, SlapR
                 sop->s_inuse = 1;                  sop->s_inuse = 1;
   
                 ldap_pvt_thread_mutex_lock( &si->si_ops_mutex );                  ldap_pvt_thread_mutex_lock( &si->si_ops_mutex );
                 while ( si->si_active ) {  
                         /* Wait for active mods to finish before proceeding, as they  
                          * may already have inspected the si_ops list looking for  
                          * consumers to replicate the change to.  Using the log  
                          * doesn't help, as we may finish playing it before the  
                          * active mods gets added to it.  
                          */  
                         ldap_pvt_thread_mutex_unlock( &si->si_ops_mutex );  
                         if ( slapd_shutdown )  
                                 return SLAPD_ABANDON;  
                         if ( !ldap_pvt_thread_pool_pausecheck( &connection_pool ))  
                                 ldap_pvt_thread_yield();  
                         ldap_pvt_thread_mutex_lock( &si->si_ops_mutex );  
                 }  
                 sop->s_next = si->si_ops;                  sop->s_next = si->si_ops;
                 si->si_ops = sop;                  si->si_ops = sop;
                 ldap_pvt_thread_mutex_unlock( &si->si_ops_mutex );                  ldap_pvt_thread_mutex_unlock( &si->si_ops_mutex );
Line 2454  syncprov_op_search( Operation *op, SlapR Line 2377  syncprov_op_search( Operation *op, SlapR
                 ctxcsn = NULL;                  ctxcsn = NULL;
                 sids = NULL;                  sids = NULL;
         }          }
         dirty = si->si_dirty;  
         ldap_pvt_thread_rdwr_runlock( &si->si_csn_rwlock );          ldap_pvt_thread_rdwr_runlock( &si->si_csn_rwlock );
                   
         /* If we have a cookie, handle the PRESENT lookups */          /* If we have a cookie, handle the PRESENT lookups */
Line 2511  syncprov_op_search( Operation *op, SlapR Line 2433  syncprov_op_search( Operation *op, SlapR
                                         if ( newer < 0 )                                          if ( newer < 0 )
                                                 changed = SS_CHANGED;                                                  changed = SS_CHANGED;
                                         else if ( newer > 0 ) {                                          else if ( newer > 0 ) {
                                         /* our state is older, complain to consumer */                                          /* our state is older, tell consumer nothing */
                                                 rs->sr_err = LDAP_UNWILLING_TO_PERFORM;  
                                                 rs->sr_text = "consumer state is newer than provider!";  
 bailout:  
                                                 if ( sop ) {                                                  if ( sop ) {
                                                         syncops **sp = &si->si_ops;                                                          syncops **sp = &si->si_ops;
                                                                                                                   
Line 2525  bailout: Line 2444  bailout:
                                                         ldap_pvt_thread_mutex_unlock( &si->si_ops_mutex );                                                          ldap_pvt_thread_mutex_unlock( &si->si_ops_mutex );
                                                         ch_free( sop );                                                          ch_free( sop );
                                                 }                                                  }
                                                   rs->sr_err = LDAP_SUCCESS;
                                                 rs->sr_ctrls = NULL;                                                  rs->sr_ctrls = NULL;
                                                 send_ldap_result( op, rs );                                                  send_ldap_result( op, rs );
                                                 return rs->sr_err;                                                  return rs->sr_err;
Line 2534  bailout: Line 2454  bailout:
                                 if ( changed )                                  if ( changed )
                                         break;                                          break;
                         }                          }
                         if ( !changed && !dirty ) {                          if ( !changed ) {
                                 do_present = 0;                                  do_present = 0;
 no_change:              if ( !(op->o_sync_mode & SLAP_SYNC_PERSIST) ) {  no_change:              if ( !(op->o_sync_mode & SLAP_SYNC_PERSIST) ) {
                                         LDAPControl     *ctrls[2];                                          LDAPControl     *ctrls[2];
Line 2579  no_change:  if ( !(op->o_sync_mode & SLA Line 2499  no_change:  if ( !(op->o_sync_mode & SLA
                                         ber_bvarray_free_x( ctxcsn, op->o_tmpmemctx );                                          ber_bvarray_free_x( ctxcsn, op->o_tmpmemctx );
                                 if ( sids )                                  if ( sids )
                                         op->o_tmpfree( sids, op->o_tmpmemctx );                                          op->o_tmpfree( sids, op->o_tmpmemctx );
                                 rs->sr_err = LDAP_SYNC_REFRESH_REQUIRED;                                  send_ldap_error( op, rs, LDAP_SYNC_REFRESH_REQUIRED, "sync cookie is stale" );
                                 rs->sr_text = "sync cookie is stale";                                  return rs->sr_err;
                                 goto bailout;  
                         }                          }
                         if ( srs->sr_state.ctxcsn ) {                          if ( srs->sr_state.ctxcsn ) {
                                 ber_bvarray_free_x( srs->sr_state.ctxcsn, op->o_tmpmemctx );                                  ber_bvarray_free_x( srs->sr_state.ctxcsn, op->o_tmpmemctx );
Line 2601  no_change:  if ( !(op->o_sync_mode & SLA Line 2520  no_change:  if ( !(op->o_sync_mode & SLA
                                         ber_bvarray_free_x( ctxcsn, op->o_tmpmemctx );                                          ber_bvarray_free_x( ctxcsn, op->o_tmpmemctx );
                                 if ( sids )                                  if ( sids )
                                         op->o_tmpfree( sids, op->o_tmpmemctx );                                          op->o_tmpfree( sids, op->o_tmpmemctx );
                                 goto bailout;                                  send_ldap_result( op, rs );
                                   return rs->sr_err;
                         }                          }
                 }                  }
         } else {          } else {
Line 2618  shortcut: Line 2538  shortcut:
         }          }
   
         /* If something changed, find the changes */          /* If something changed, find the changes */
         if ( gotstate && ( changed || dirty ) ) {          if ( gotstate && changed ) {
                 Filter *fand, *fava;                  Filter *fand, *fava;
   
                 fand = op->o_tmpalloc( sizeof(Filter), op->o_tmpmemctx );                  fand = op->o_tmpalloc( sizeof(Filter), op->o_tmpmemctx );
Line 2634  shortcut: Line 2554  shortcut:
 #endif  #endif
                 ber_dupbv_x( &fava->f_ava->aa_value, &mincsn, op->o_tmpmemctx );                  ber_dupbv_x( &fava->f_ava->aa_value, &mincsn, op->o_tmpmemctx );
                 fava->f_next = op->ors_filter;                  fava->f_next = op->ors_filter;
                 if ( sop )  
                         ldap_pvt_thread_mutex_lock( &sop->s_mutex );  
                 op->ors_filter = fand;                  op->ors_filter = fand;
                 filter2bv_x( op, op->ors_filter, &op->ors_filterstr );                  filter2bv_x( op, op->ors_filter, &op->ors_filterstr );
                 if ( sop ) {                  if ( sop )
                         sop->s_flags |= PS_FIX_FILTER;                          sop->s_flags |= PS_FIX_FILTER;
                         ldap_pvt_thread_mutex_unlock( &sop->s_mutex );  
                 }  
         }          }
   
         /* Let our callback add needed info to returned entries */          /* Let our callback add needed info to returned entries */
Line 2663  shortcut: Line 2579  shortcut:
          * the refresh phase, just invoke the response callback to transition           * the refresh phase, just invoke the response callback to transition
          * us into persist phase           * us into persist phase
          */           */
         if ( !changed && !dirty ) {          if ( !changed ) {
                 rs->sr_err = LDAP_SUCCESS;                  rs->sr_err = LDAP_SUCCESS;
                 rs->sr_nentries = 0;                  rs->sr_nentries = 0;
                 send_ldap_result( op, rs );                  send_ldap_result( op, rs );
Line 2974  syncprov_db_open( Line 2890  syncprov_db_open(
         }          }
   
         thrctx = ldap_pvt_thread_pool_context();          thrctx = ldap_pvt_thread_pool_context();
         connection_fake_init2( &conn, &opbuf, thrctx, 0 );          connection_fake_init( &conn, &opbuf, thrctx );
         op = &opbuf.ob_op;          op = &opbuf.ob_op;
         op->o_bd = be;          op->o_bd = be;
         op->o_dn = be->be_rootdn;          op->o_dn = be->be_rootdn;
Line 3059  syncprov_db_close( Line 2975  syncprov_db_close(
                 void *thrctx;                  void *thrctx;
   
                 thrctx = ldap_pvt_thread_pool_context();                  thrctx = ldap_pvt_thread_pool_context();
                 connection_fake_init2( &conn, &opbuf, thrctx, 0 );                  connection_fake_init( &conn, &opbuf, thrctx );
                 op = &opbuf.ob_op;                  op = &opbuf.ob_op;
                 op->o_bd = be;                  op->o_bd = be;
                 op->o_dn = be->be_rootdn;                  op->o_dn = be->be_rootdn;

Removed from v.1.147.2.79  
changed lines
  Added in v.1.293


______________
© Copyright 1998-2020, OpenLDAP Foundation, info@OpenLDAP.org