2000-09-20 04:13:41 +08:00
|
|
|
/* init.c - initialize bdb backend */
|
|
|
|
/* $OpenLDAP$ */
|
|
|
|
/*
|
|
|
|
* Copyright 1998-2000 The OpenLDAP Foundation, All Rights Reserved.
|
|
|
|
* COPYING RESTRICTIONS APPLY, see COPYRIGHT file
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "portable.h"
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <ac/string.h>
|
|
|
|
|
|
|
|
#include "back-bdb.h"
|
|
|
|
|
|
|
|
int bdb_next_id( BackendDB *be, DB_TXN *tid, ID *out )
|
|
|
|
{
|
|
|
|
struct bdb_info *bdb = (struct bdb_info *) be->be_private;
|
|
|
|
int rc;
|
|
|
|
ID kid = NOID;
|
|
|
|
ID id;
|
|
|
|
DBT key, data;
|
2000-09-22 09:40:57 +08:00
|
|
|
DB_TXN *ltid;
|
|
|
|
|
2000-09-20 04:13:41 +08:00
|
|
|
DBTzero( &key );
|
|
|
|
key.data = (char *) &kid;
|
|
|
|
key.size = sizeof( kid );
|
|
|
|
|
|
|
|
DBTzero( &data );
|
|
|
|
data.data = (char *) &id;
|
|
|
|
data.ulen = sizeof( id );
|
|
|
|
data.flags = DB_DBT_USERMEM;
|
|
|
|
|
2000-09-24 07:15:40 +08:00
|
|
|
if( 0 ) {
|
|
|
|
retry: if( tid != NULL ) {
|
|
|
|
/* nested transaction, abort and return */
|
|
|
|
(void) txn_abort( ltid );
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
rc = txn_abort( ltid );
|
|
|
|
if( rc != 0 ) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = txn_begin( bdb->bi_dbenv, tid, <id, 0 );
|
|
|
|
if( rc != 0 ) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* get existing value for read/modify/write */
|
2000-09-22 14:46:32 +08:00
|
|
|
rc = bdb->bi_nextid->bdi_db->get( bdb->bi_nextid->bdi_db,
|
2000-09-22 09:40:57 +08:00
|
|
|
ltid, &key, &data, DB_RMW );
|
2000-09-20 04:13:41 +08:00
|
|
|
|
2000-09-24 07:15:40 +08:00
|
|
|
switch(rc) {
|
|
|
|
case DB_LOCK_DEADLOCK:
|
|
|
|
case DB_LOCK_NOTGRANTED:
|
|
|
|
goto retry;
|
|
|
|
|
|
|
|
case DB_NOTFOUND:
|
2000-09-20 04:13:41 +08:00
|
|
|
id = NOID;
|
2000-09-24 07:15:40 +08:00
|
|
|
break;
|
2000-09-20 04:13:41 +08:00
|
|
|
|
2000-09-24 07:15:40 +08:00
|
|
|
case 0:
|
|
|
|
if ( data.size != sizeof(ID) ) {
|
|
|
|
/* size mismatch! */
|
|
|
|
rc = -1;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
break;
|
2000-09-20 04:13:41 +08:00
|
|
|
|
2000-09-24 07:15:40 +08:00
|
|
|
default:
|
2000-09-22 09:40:57 +08:00
|
|
|
goto done;
|
2000-09-20 04:13:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
id++;
|
|
|
|
|
2000-09-24 07:15:40 +08:00
|
|
|
/* put new value */
|
2000-09-22 14:46:32 +08:00
|
|
|
rc = bdb->bi_nextid->bdi_db->put( bdb->bi_nextid->bdi_db,
|
2000-09-22 09:40:57 +08:00
|
|
|
ltid, &key, &data, 0 );
|
2000-09-20 04:13:41 +08:00
|
|
|
|
2000-09-24 07:15:40 +08:00
|
|
|
switch(rc) {
|
|
|
|
case DB_LOCK_DEADLOCK:
|
|
|
|
case DB_LOCK_NOTGRANTED:
|
|
|
|
goto retry;
|
2000-09-22 09:40:57 +08:00
|
|
|
|
2000-09-24 07:15:40 +08:00
|
|
|
case 0:
|
|
|
|
*out = id;
|
2000-09-22 09:40:57 +08:00
|
|
|
rc = txn_commit( ltid, 0 );
|
2000-09-24 07:15:40 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
done: (void) txn_abort( ltid );
|
2000-09-22 09:40:57 +08:00
|
|
|
}
|
|
|
|
|
2000-09-20 04:13:41 +08:00
|
|
|
return rc;
|
|
|
|
}
|