2009-08-16 12:44:06 +04:00
/*
Unix SMB / CIFS implementation .
Little dictionary style data structure based on dbwrap_rbt
Copyright ( C ) Volker Lendecke 2009
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 3 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
2011-07-07 19:42:08 +04:00
# include "dbwrap/dbwrap.h"
2011-08-16 18:20:14 +04:00
# include "dbwrap/dbwrap_rbt.h"
2010-08-18 21:05:59 +04:00
# include "talloc_dict.h"
2011-05-05 13:25:29 +04:00
# include "util_tdb.h"
2009-08-16 12:44:06 +04:00
struct talloc_dict {
struct db_context * db ;
} ;
struct talloc_dict * talloc_dict_init ( TALLOC_CTX * mem_ctx )
{
struct talloc_dict * result ;
result = talloc ( mem_ctx , struct talloc_dict ) ;
if ( result = = NULL ) {
return NULL ;
}
result - > db = db_open_rbt ( result ) ;
if ( result - > db = = NULL ) {
TALLOC_FREE ( result ) ;
return NULL ;
}
return result ;
}
/*
* Add a talloced object to the dict . Nulls out the pointer to indicate that
* the talloc ownership has been taken . If an object for " key " already exists ,
* the existing object is talloc_free ( ) ed and overwritten by the new
* object . If " data " is NULL , object for key " key " is deleted . Return false
* for " no memory " .
*/
bool talloc_dict_set ( struct talloc_dict * dict , DATA_BLOB key , void * pdata )
{
struct db_record * rec ;
NTSTATUS status = NT_STATUS_OK ;
void * data = * ( void * * ) pdata ;
2011-08-17 12:50:31 +04:00
TDB_DATA value ;
2009-08-16 12:44:06 +04:00
2011-08-17 12:50:31 +04:00
rec = dbwrap_fetch_locked ( dict - > db , talloc_tos ( ) ,
make_tdb_data ( key . data , key . length ) ) ;
2009-08-16 12:44:06 +04:00
if ( rec = = NULL ) {
return false ;
}
2011-08-17 12:50:31 +04:00
value = dbwrap_record_get_value ( rec ) ;
if ( value . dsize ! = 0 ) {
2009-08-16 12:44:06 +04:00
void * old_data ;
2011-08-17 12:50:31 +04:00
if ( value . dsize ! = sizeof ( void * ) ) {
2009-08-16 12:44:06 +04:00
TALLOC_FREE ( rec ) ;
return false ;
}
2011-08-17 12:50:31 +04:00
old_data = * ( void * * ) ( value . dptr ) ;
2009-08-16 12:44:06 +04:00
TALLOC_FREE ( old_data ) ;
if ( data = = NULL ) {
2011-08-17 12:50:31 +04:00
status = dbwrap_record_delete ( rec ) ;
2009-08-16 12:44:06 +04:00
}
}
if ( data ! = NULL ) {
void * mydata = talloc_move ( dict - > db , & data ) ;
* ( void * * ) pdata = NULL ;
2011-08-17 12:50:31 +04:00
status = dbwrap_record_store ( rec ,
make_tdb_data ( ( uint8_t * ) & mydata ,
sizeof ( mydata ) ) , 0 ) ;
2009-08-16 12:44:06 +04:00
}
TALLOC_FREE ( rec ) ;
return NT_STATUS_IS_OK ( status ) ;
}
/*
* Fetch a talloced object . If " mem_ctx!=NULL " , talloc_move the object there
* and delete it from the dict .
*/
void * talloc_dict_fetch ( struct talloc_dict * dict , DATA_BLOB key ,
TALLOC_CTX * mem_ctx )
{
struct db_record * rec ;
void * result ;
2011-08-17 12:50:31 +04:00
TDB_DATA value ;
2009-08-16 12:44:06 +04:00
2011-08-17 12:50:31 +04:00
rec = dbwrap_fetch_locked ( dict - > db , talloc_tos ( ) ,
make_tdb_data ( key . data , key . length ) ) ;
2009-08-16 12:44:06 +04:00
if ( rec = = NULL ) {
return NULL ;
}
2011-08-17 12:50:31 +04:00
value = dbwrap_record_get_value ( rec ) ;
if ( value . dsize ! = sizeof ( void * ) ) {
2009-08-16 12:44:06 +04:00
TALLOC_FREE ( rec ) ;
return NULL ;
}
2011-08-17 12:50:31 +04:00
result = * ( void * * ) value . dptr ;
2009-08-16 12:44:06 +04:00
if ( mem_ctx ! = NULL ) {
NTSTATUS status ;
2011-08-17 12:50:31 +04:00
status = dbwrap_record_delete ( rec ) ;
2009-08-16 12:44:06 +04:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
TALLOC_FREE ( rec ) ;
return NULL ;
}
result = talloc_move ( mem_ctx , & result ) ;
}
return result ;
}
struct talloc_dict_traverse_state {
int ( * fn ) ( DATA_BLOB key , void * data , void * private_data ) ;
void * private_data ;
} ;
static int talloc_dict_traverse_fn ( struct db_record * rec , void * private_data )
{
2011-08-17 12:50:31 +04:00
TDB_DATA key ;
TDB_DATA value ;
2009-08-16 12:44:06 +04:00
struct talloc_dict_traverse_state * state =
( struct talloc_dict_traverse_state * ) private_data ;
2011-08-17 12:50:31 +04:00
key = dbwrap_record_get_key ( rec ) ;
value = dbwrap_record_get_value ( rec ) ;
if ( value . dsize ! = sizeof ( void * ) ) {
2009-08-16 12:44:06 +04:00
return - 1 ;
}
2011-08-17 12:50:31 +04:00
return state - > fn ( data_blob_const ( key . dptr , key . dsize ) ,
* ( void * * ) value . dptr , state - > private_data ) ;
2009-08-16 12:44:06 +04:00
}
/*
* Traverse a talloc_dict . If " fn " returns non - null , quit the traverse
*/
int talloc_dict_traverse ( struct talloc_dict * dict ,
int ( * fn ) ( DATA_BLOB key , void * data ,
void * private_data ) ,
void * private_data )
{
struct talloc_dict_traverse_state state ;
2011-08-17 12:50:31 +04:00
NTSTATUS status ;
int count = 0 ;
2009-08-16 12:44:06 +04:00
state . fn = fn ;
state . private_data = private_data ;
2011-08-17 12:50:31 +04:00
status = dbwrap_traverse ( dict - > db , talloc_dict_traverse_fn , & state ,
& count ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return - 1 ;
} else {
return count ;
}
2009-08-16 12:44:06 +04:00
}