2007-11-10 01:43:24 +03:00
/*
Unix SMB / CIFS implementation .
Database interface wrapper around red - black trees
2008-01-12 12:38:17 +03:00
Copyright ( C ) Volker Lendecke 2007 , 2008
2007-11-10 01:43:24 +03:00
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-24 16:53:42 +04:00
# include "dbwrap/dbwrap_private.h"
2011-08-16 18:20:14 +04:00
# include "dbwrap/dbwrap_rbt.h"
2008-10-12 18:27:00 +04:00
# include "../lib/util/rbtree.h"
2007-11-10 01:43:24 +03:00
2008-06-19 11:50:34 +04:00
# define DBWRAP_RBT_ALIGN(_size_) (((_size_)+15)&~15)
2008-01-12 12:38:17 +03:00
2007-11-10 01:43:24 +03:00
struct db_rbt_ctx {
struct rb_root tree ;
} ;
struct db_rbt_rec {
struct db_rbt_node * node ;
} ;
/* The structure that ends up in the tree */
struct db_rbt_node {
struct rb_node rb_node ;
size_t keysize , valuesize ;
/*
* key and value are appended implicitly , " data " is only here as a
* target for offsetof ( )
*/
2008-06-19 11:50:34 +04:00
char data [ 1 ] ;
2007-11-10 01:43:24 +03:00
} ;
2008-01-12 12:38:17 +03:00
/*
* Hide the ugly pointer calculations in a function
*/
static struct db_rbt_node * db_rbt2node ( struct rb_node * node )
{
return ( struct db_rbt_node * )
( ( char * ) node - offsetof ( struct db_rbt_node , rb_node ) ) ;
}
/*
* Compare two keys
*/
static int db_rbt_compare ( TDB_DATA a , TDB_DATA b )
{
int res ;
res = memcmp ( a . dptr , b . dptr , MIN ( a . dsize , b . dsize ) ) ;
if ( ( res < 0 ) | | ( ( res = = 0 ) & & ( a . dsize < b . dsize ) ) ) {
return - 1 ;
}
if ( ( res > 0 ) | | ( ( res = = 0 ) & & ( a . dsize > b . dsize ) ) ) {
return 1 ;
}
return 0 ;
}
2007-11-10 01:43:24 +03:00
/*
* dissect a db_rbt_node into its implicit key and value parts
*/
static void db_rbt_parse_node ( struct db_rbt_node * node ,
TDB_DATA * key , TDB_DATA * value )
{
2012-05-11 23:54:46 +04:00
key - > dptr = ( ( uint8_t * ) node ) + offsetof ( struct db_rbt_node , data ) ;
2007-11-10 01:43:24 +03:00
key - > dsize = node - > keysize ;
value - > dptr = key - > dptr + node - > keysize ;
value - > dsize = node - > valuesize ;
}
static NTSTATUS db_rbt_store ( struct db_record * rec , TDB_DATA data , int flag )
{
2012-05-30 18:05:03 +04:00
struct db_rbt_ctx * db_ctx = talloc_get_type_abort (
rec - > db - > private_data , struct db_rbt_ctx ) ;
2008-01-12 12:38:17 +03:00
struct db_rbt_rec * rec_priv = ( struct db_rbt_rec * ) rec - > private_data ;
2007-11-10 01:43:24 +03:00
struct db_rbt_node * node ;
struct rb_node * * p ;
struct rb_node * parent ;
TDB_DATA this_key , this_val ;
if ( rec_priv - > node ! = NULL ) {
/*
* The record was around previously
*/
db_rbt_parse_node ( rec_priv - > node , & this_key , & this_val ) ;
SMB_ASSERT ( this_key . dsize = = rec - > key . dsize ) ;
SMB_ASSERT ( memcmp ( this_key . dptr , rec - > key . dptr ,
this_key . dsize ) = = 0 ) ;
if ( this_val . dsize > = data . dsize ) {
/*
* The new value fits into the old space
*/
memcpy ( this_val . dptr , data . dptr , data . dsize ) ;
rec_priv - > node - > valuesize = data . dsize ;
return NT_STATUS_OK ;
}
2012-05-30 18:50:06 +04:00
}
node = ( struct db_rbt_node * ) talloc_size ( db_ctx ,
offsetof ( struct db_rbt_node , data ) + rec - > key . dsize
+ data . dsize ) ;
if ( node = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
2007-11-10 01:43:24 +03:00
2012-05-30 18:50:06 +04:00
if ( rec_priv - > node ! = NULL ) {
2007-11-10 01:43:24 +03:00
/*
* We need to delete the key from the tree and start fresh ,
* there ' s not enough space in the existing record
*/
2012-05-30 18:05:03 +04:00
rb_erase ( & rec_priv - > node - > rb_node , & db_ctx - > tree ) ;
2007-12-19 15:48:49 +03:00
/*
* Keep the existing node around for a while : If the record
* existed before , we reference the key data in there .
*/
2007-11-10 01:43:24 +03:00
}
ZERO_STRUCT ( node - > rb_node ) ;
node - > keysize = rec - > key . dsize ;
node - > valuesize = data . dsize ;
db_rbt_parse_node ( node , & this_key , & this_val ) ;
memcpy ( this_key . dptr , rec - > key . dptr , node - > keysize ) ;
2009-02-27 17:28:52 +03:00
TALLOC_FREE ( rec_priv - > node ) ;
2012-05-30 18:48:39 +04:00
rec_priv - > node = node ;
2007-11-10 01:43:24 +03:00
2007-12-19 15:48:49 +03:00
memcpy ( this_val . dptr , data . dptr , node - > valuesize ) ;
2007-12-19 04:30:02 +03:00
2007-11-10 01:43:24 +03:00
parent = NULL ;
2012-05-30 18:05:03 +04:00
p = & db_ctx - > tree . rb_node ;
2007-11-10 01:43:24 +03:00
while ( * p ) {
struct db_rbt_node * r ;
TDB_DATA search_key , search_val ;
int res ;
parent = ( * p ) ;
2008-01-12 12:38:17 +03:00
r = db_rbt2node ( * p ) ;
2007-11-10 01:43:24 +03:00
db_rbt_parse_node ( r , & search_key , & search_val ) ;
2008-01-12 12:38:17 +03:00
res = db_rbt_compare ( this_key , search_key ) ;
2007-11-10 01:43:24 +03:00
2008-01-12 12:38:17 +03:00
if ( res = = - 1 ) {
2007-11-10 01:43:24 +03:00
p = & ( * p ) - > rb_left ;
}
2008-01-12 12:38:17 +03:00
else if ( res = = 1 ) {
2007-11-10 01:43:24 +03:00
p = & ( * p ) - > rb_right ;
}
else {
smb_panic ( " someone messed with the tree " ) ;
}
}
rb_link_node ( & node - > rb_node , parent , p ) ;
2012-05-30 18:05:03 +04:00
rb_insert_color ( & node - > rb_node , & db_ctx - > tree ) ;
2007-11-10 01:43:24 +03:00
return NT_STATUS_OK ;
}
static NTSTATUS db_rbt_delete ( struct db_record * rec )
{
2012-05-30 18:05:03 +04:00
struct db_rbt_ctx * db_ctx = talloc_get_type_abort (
rec - > db - > private_data , struct db_rbt_ctx ) ;
2008-01-12 12:38:17 +03:00
struct db_rbt_rec * rec_priv = ( struct db_rbt_rec * ) rec - > private_data ;
2007-11-10 01:43:24 +03:00
if ( rec_priv - > node = = NULL ) {
return NT_STATUS_OK ;
}
2012-05-30 18:05:03 +04:00
rb_erase ( & rec_priv - > node - > rb_node , & db_ctx - > tree ) ;
2009-02-27 17:28:52 +03:00
TALLOC_FREE ( rec_priv - > node ) ;
2007-11-10 01:43:24 +03:00
return NT_STATUS_OK ;
}
2012-05-30 17:06:12 +04:00
static NTSTATUS db_rbt_store_deny ( struct db_record * rec , TDB_DATA data , int flag )
{
return NT_STATUS_MEDIA_WRITE_PROTECTED ;
}
static NTSTATUS db_rbt_delete_deny ( struct db_record * rec )
{
return NT_STATUS_MEDIA_WRITE_PROTECTED ;
}
2011-07-04 11:54:31 +04:00
struct db_rbt_search_result {
TDB_DATA key ;
TDB_DATA val ;
struct db_rbt_node * node ;
} ;
2007-11-10 01:43:24 +03:00
2011-07-04 11:54:31 +04:00
static bool db_rbt_search_internal ( struct db_context * db , TDB_DATA key ,
struct db_rbt_search_result * result )
2007-11-10 01:43:24 +03:00
{
struct db_rbt_ctx * ctx = talloc_get_type_abort (
2011-07-04 11:54:31 +04:00
db - > private_data , struct db_rbt_ctx ) ;
2007-11-10 01:43:24 +03:00
struct rb_node * n ;
2008-01-12 12:38:17 +03:00
bool found = false ;
struct db_rbt_node * r = NULL ;
2011-07-04 11:54:31 +04:00
TDB_DATA search_key , search_val ;
2007-11-10 01:43:24 +03:00
2008-01-12 12:38:17 +03:00
n = ctx - > tree . rb_node ;
2007-11-10 01:43:24 +03:00
2008-01-12 12:38:17 +03:00
while ( n ! = NULL ) {
int res ;
r = db_rbt2node ( n ) ;
db_rbt_parse_node ( r , & search_key , & search_val ) ;
res = db_rbt_compare ( key , search_key ) ;
if ( res = = - 1 ) {
n = n - > rb_left ;
}
else if ( res = = 1 ) {
n = n - > rb_right ;
}
else {
found = true ;
break ;
}
2007-11-10 01:43:24 +03:00
}
2011-07-04 11:54:31 +04:00
if ( result ! = NULL ) {
if ( found ) {
result - > key = search_key ;
result - > val = search_val ;
result - > node = r ;
} else {
ZERO_STRUCT ( * result ) ;
}
}
return found ;
}
static struct db_record * db_rbt_fetch_locked ( struct db_context * db_ctx ,
TALLOC_CTX * mem_ctx ,
TDB_DATA key )
{
struct db_rbt_rec * rec_priv ;
struct db_record * result ;
size_t size ;
bool found ;
struct db_rbt_search_result res ;
found = db_rbt_search_internal ( db_ctx , key , & res ) ;
2007-11-10 01:43:24 +03:00
2008-01-12 12:38:17 +03:00
/*
* In this low - level routine , play tricks to reduce the number of
* tallocs to one . Not recommened for general use , but here it pays
* off .
*/
2007-11-10 01:43:24 +03:00
2008-06-19 11:50:34 +04:00
size = DBWRAP_RBT_ALIGN ( sizeof ( struct db_record ) )
+ sizeof ( struct db_rbt_rec ) ;
2008-01-12 12:38:17 +03:00
if ( ! found ) {
/*
* We need to keep the key around for later store
*/
size + = key . dsize ;
}
result = ( struct db_record * ) talloc_size ( mem_ctx , size ) ;
if ( result = = NULL ) {
2007-11-10 01:43:24 +03:00
return NULL ;
}
2008-01-12 12:38:17 +03:00
rec_priv = ( struct db_rbt_rec * )
2008-06-19 11:50:34 +04:00
( ( char * ) result + DBWRAP_RBT_ALIGN ( sizeof ( struct db_record ) ) ) ;
2007-11-10 01:43:24 +03:00
result - > store = db_rbt_store ;
result - > delete_rec = db_rbt_delete ;
result - > private_data = rec_priv ;
2011-07-04 11:54:31 +04:00
rec_priv - > node = res . node ;
result - > value = res . val ;
2008-01-12 12:38:17 +03:00
if ( found ) {
2011-07-04 11:54:31 +04:00
result - > key = res . key ;
2008-01-12 12:38:17 +03:00
}
else {
2012-05-11 23:54:46 +04:00
result - > key . dptr = ( uint8_t * )
2008-01-12 12:38:17 +03:00
( ( char * ) rec_priv + sizeof ( * rec_priv ) ) ;
result - > key . dsize = key . dsize ;
memcpy ( result - > key . dptr , key . dptr , key . dsize ) ;
}
return result ;
}
2011-07-04 12:22:46 +04:00
static int db_rbt_exists ( struct db_context * db , TDB_DATA key )
{
return db_rbt_search_internal ( db , key , NULL ) ;
}
2011-08-16 16:55:30 +04:00
static int db_rbt_wipe ( struct db_context * db )
{
struct db_rbt_ctx * old_ctx = talloc_get_type_abort (
db - > private_data , struct db_rbt_ctx ) ;
2011-08-17 13:44:12 +04:00
struct db_rbt_ctx * new_ctx = talloc_zero ( db , struct db_rbt_ctx ) ;
2011-08-16 16:55:30 +04:00
if ( new_ctx = = NULL ) {
return - 1 ;
}
db - > private_data = new_ctx ;
talloc_free ( old_ctx ) ;
return 0 ;
}
2011-12-08 18:50:33 +04:00
static NTSTATUS db_rbt_parse_record ( struct db_context * db , TDB_DATA key ,
void ( * parser ) ( TDB_DATA key , TDB_DATA data ,
void * private_data ) ,
void * private_data )
2011-07-05 17:06:05 +04:00
{
struct db_rbt_search_result res ;
bool found = db_rbt_search_internal ( db , key , & res ) ;
if ( ! found ) {
2011-12-08 18:50:33 +04:00
return NT_STATUS_NOT_FOUND ;
2011-07-05 17:06:05 +04:00
}
2011-12-08 18:50:33 +04:00
parser ( res . key , res . val , private_data ) ;
return NT_STATUS_OK ;
2011-07-05 17:06:05 +04:00
}
2012-05-30 17:06:12 +04:00
static int db_rbt_traverse_internal ( struct db_context * db ,
struct rb_node * n ,
2009-08-08 15:43:41 +04:00
int ( * f ) ( struct db_record * db ,
void * private_data ) ,
2012-05-30 17:06:12 +04:00
void * private_data , uint32_t * count ,
bool rw )
2009-08-08 15:43:41 +04:00
{
2012-05-30 17:06:12 +04:00
struct rb_node * rb_right ;
struct rb_node * rb_left ;
2009-08-08 15:43:41 +04:00
struct db_record rec ;
2012-05-30 17:06:12 +04:00
struct db_rbt_rec rec_priv ;
2009-08-08 15:43:41 +04:00
int ret ;
if ( n = = NULL ) {
return 0 ;
}
2012-05-30 17:06:12 +04:00
rb_left = n - > rb_left ;
rb_right = n - > rb_right ;
ret = db_rbt_traverse_internal ( db , rb_left , f , private_data , count , rw ) ;
2009-08-08 15:43:41 +04:00
if ( ret ! = 0 ) {
return ret ;
}
2012-05-30 17:06:12 +04:00
rec_priv . node = db_rbt2node ( n ) ;
/* n might be altered by the callback function */
n = NULL ;
2009-08-08 15:43:41 +04:00
ZERO_STRUCT ( rec ) ;
2012-05-30 17:06:12 +04:00
rec . db = db ;
rec . private_data = & rec_priv ;
if ( rw ) {
rec . store = db_rbt_store ;
rec . delete_rec = db_rbt_delete ;
} else {
rec . store = db_rbt_store_deny ;
rec . delete_rec = db_rbt_delete_deny ;
}
db_rbt_parse_node ( rec_priv . node , & rec . key , & rec . value ) ;
2009-08-08 15:43:41 +04:00
ret = f ( & rec , private_data ) ;
2011-08-17 13:44:12 +04:00
( * count ) + + ;
2009-08-08 15:43:41 +04:00
if ( ret ! = 0 ) {
return ret ;
}
2012-05-30 17:06:12 +04:00
if ( rec_priv . node ! = NULL ) {
/*
* If the current record is still there
* we should take the current rb_right .
*/
rb_right = rec_priv . node - > rb_node . rb_right ;
}
return db_rbt_traverse_internal ( db , rb_right , f , private_data , count , rw ) ;
2009-08-08 15:43:41 +04:00
}
2007-11-10 01:43:24 +03:00
static int db_rbt_traverse ( struct db_context * db ,
int ( * f ) ( struct db_record * db ,
void * private_data ) ,
void * private_data )
{
2009-08-08 15:43:41 +04:00
struct db_rbt_ctx * ctx = talloc_get_type_abort (
db - > private_data , struct db_rbt_ctx ) ;
2011-08-17 13:44:12 +04:00
uint32_t count = 0 ;
2009-08-08 15:43:41 +04:00
2012-05-30 17:06:12 +04:00
int ret = db_rbt_traverse_internal ( db , ctx - > tree . rb_node ,
f , private_data , & count ,
true /* rw */ ) ;
if ( ret ! = 0 ) {
return - 1 ;
}
if ( count > INT_MAX ) {
return - 1 ;
}
return count ;
}
static int db_rbt_traverse_read ( struct db_context * db ,
int ( * f ) ( struct db_record * db ,
void * private_data ) ,
void * private_data )
{
struct db_rbt_ctx * ctx = talloc_get_type_abort (
db - > private_data , struct db_rbt_ctx ) ;
uint32_t count = 0 ;
int ret = db_rbt_traverse_internal ( db , ctx - > tree . rb_node ,
f , private_data , & count ,
false /* rw */ ) ;
2011-08-17 13:44:12 +04:00
if ( ret ! = 0 ) {
return - 1 ;
}
if ( count > INT_MAX ) {
return - 1 ;
}
return count ;
2007-11-10 01:43:24 +03:00
}
static int db_rbt_get_seqnum ( struct db_context * db )
{
return 0 ;
}
2008-03-10 12:17:05 +03:00
static int db_rbt_trans_dummy ( struct db_context * db )
{
/*
* Transactions are pretty pointless in - memory , just return success .
*/
return 0 ;
}
2012-02-15 17:57:01 +04:00
static void db_rbt_id ( struct db_context * db , const uint8_t * * id , size_t * idlen )
{
* id = ( uint8_t * ) db ;
2012-05-15 20:51:06 +04:00
* idlen = sizeof ( struct db_context * ) ;
2012-02-15 17:57:01 +04:00
}
2007-11-10 01:43:24 +03:00
struct db_context * db_open_rbt ( TALLOC_CTX * mem_ctx )
{
struct db_context * result ;
result = talloc ( mem_ctx , struct db_context ) ;
if ( result = = NULL ) {
return NULL ;
}
2011-06-07 05:44:43 +04:00
result - > private_data = talloc_zero ( result , struct db_rbt_ctx ) ;
2007-11-10 01:43:24 +03:00
if ( result - > private_data = = NULL ) {
TALLOC_FREE ( result ) ;
return NULL ;
}
result - > fetch_locked = db_rbt_fetch_locked ;
2012-03-27 16:31:04 +04:00
result - > try_fetch_locked = NULL ;
2007-11-10 01:43:24 +03:00
result - > traverse = db_rbt_traverse ;
2012-05-30 17:06:12 +04:00
result - > traverse_read = db_rbt_traverse_read ;
2007-11-10 01:43:24 +03:00
result - > get_seqnum = db_rbt_get_seqnum ;
2008-03-10 12:17:05 +03:00
result - > transaction_start = db_rbt_trans_dummy ;
result - > transaction_commit = db_rbt_trans_dummy ;
result - > transaction_cancel = db_rbt_trans_dummy ;
2011-07-04 12:22:46 +04:00
result - > exists = db_rbt_exists ;
2011-08-16 16:55:30 +04:00
result - > wipe = db_rbt_wipe ;
2011-07-05 17:06:05 +04:00
result - > parse_record = db_rbt_parse_record ;
2012-01-08 22:04:39 +04:00
result - > lock_order = 0 ;
2012-02-15 17:57:01 +04:00
result - > id = db_rbt_id ;
2012-06-22 09:37:44 +04:00
result - > name = " dbwrap rbt " ;
2012-06-22 09:37:44 +04:00
result - > hash_size = 0 ;
2012-02-15 18:08:29 +04:00
result - > stored_callback = NULL ;
2007-11-10 01:43:24 +03:00
return result ;
}