2008-01-16 12:09:48 +03:00
/*
Unix SMB / CIFS implementation .
Utility functions for the dbwrap API
Copyright ( C ) Volker Lendecke 2007
2009-07-15 16:59:07 +02:00
Copyright ( C ) Michael Adam 2009
2011-05-30 17:27:23 +10:00
Copyright ( C ) Jim McDonough < jmcd @ us . ibm . com > 2006
Major code contributions from Aleksey Fedoseev ( fedoseev @ ru . ibm . com )
2008-01-16 12:09:48 +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 2 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 , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include "includes.h"
2010-08-18 18:59:23 +02:00
# include "dbwrap.h"
2011-05-05 11:25:29 +02:00
# include "util_tdb.h"
2008-01-16 12:09:48 +03:00
int32_t dbwrap_fetch_int32 ( struct db_context * db , const char * keystr )
{
TDB_DATA dbuf ;
int32 ret ;
if ( db - > fetch ( db , NULL , string_term_tdb_data ( keystr ) , & dbuf ) ! = 0 ) {
return - 1 ;
}
if ( ( dbuf . dptr = = NULL ) | | ( dbuf . dsize ! = sizeof ( int32_t ) ) ) {
TALLOC_FREE ( dbuf . dptr ) ;
return - 1 ;
}
ret = IVAL ( dbuf . dptr , 0 ) ;
TALLOC_FREE ( dbuf . dptr ) ;
return ret ;
}
int dbwrap_store_int32 ( struct db_context * db , const char * keystr , int32_t v )
{
struct db_record * rec ;
int32 v_store ;
NTSTATUS status ;
rec = db - > fetch_locked ( db , NULL , string_term_tdb_data ( keystr ) ) ;
if ( rec = = NULL ) {
return - 1 ;
}
SIVAL ( & v_store , 0 , v ) ;
2008-03-28 09:42:37 +01:00
status = rec - > store ( rec , make_tdb_data ( ( const uint8 * ) & v_store ,
sizeof ( v_store ) ) ,
TDB_REPLACE ) ;
TALLOC_FREE ( rec ) ;
return NT_STATUS_IS_OK ( status ) ? 0 : - 1 ;
}
bool dbwrap_fetch_uint32 ( struct db_context * db , const char * keystr ,
uint32_t * val )
{
TDB_DATA dbuf ;
if ( db - > fetch ( db , NULL , string_term_tdb_data ( keystr ) , & dbuf ) ! = 0 ) {
return false ;
}
if ( ( dbuf . dptr = = NULL ) | | ( dbuf . dsize ! = sizeof ( uint32_t ) ) ) {
TALLOC_FREE ( dbuf . dptr ) ;
return false ;
}
* val = IVAL ( dbuf . dptr , 0 ) ;
TALLOC_FREE ( dbuf . dptr ) ;
return true ;
}
2008-12-17 17:16:50 +01:00
int dbwrap_store_uint32 ( struct db_context * db , const char * keystr , uint32_t v )
2008-03-28 09:42:37 +01:00
{
struct db_record * rec ;
uint32 v_store ;
NTSTATUS status ;
rec = db - > fetch_locked ( db , NULL , string_term_tdb_data ( keystr ) ) ;
if ( rec = = NULL ) {
2008-12-17 17:16:50 +01:00
return - 1 ;
2008-03-28 09:42:37 +01:00
}
SIVAL ( & v_store , 0 , v ) ;
2008-01-16 12:09:48 +03:00
status = rec - > store ( rec , make_tdb_data ( ( const uint8 * ) & v_store ,
sizeof ( v_store ) ) ,
TDB_REPLACE ) ;
TALLOC_FREE ( rec ) ;
return NT_STATUS_IS_OK ( status ) ? 0 : - 1 ;
}
2008-08-05 23:13:06 +02:00
/**
* Atomic unsigned integer change ( addition ) :
*
* if value does not exist yet in the db , use * oldval as initial old value .
* return old value in * oldval .
* store * oldval + change_val to db .
*/
2009-07-29 14:32:31 +02:00
struct dbwrap_change_uint32_atomic_context {
const char * keystr ;
uint32_t * oldval ;
uint32_t change_val ;
} ;
static NTSTATUS dbwrap_change_uint32_atomic_action ( struct db_context * db ,
void * private_data )
2008-01-16 12:09:48 +03:00
{
struct db_record * rec ;
2011-02-09 10:43:56 -08:00
uint32_t val = ( uint32_t ) - 1 ;
2009-11-21 00:08:38 +01:00
uint32_t v_store ;
2009-07-29 14:16:11 +02:00
NTSTATUS ret ;
2009-07-29 14:32:31 +02:00
struct dbwrap_change_uint32_atomic_context * state ;
2008-01-16 12:09:48 +03:00
2009-07-29 14:32:31 +02:00
state = ( struct dbwrap_change_uint32_atomic_context * ) private_data ;
rec = db - > fetch_locked ( db , NULL , string_term_tdb_data ( state - > keystr ) ) ;
2009-07-29 14:25:29 +02:00
if ( ! rec ) {
2009-07-29 14:16:11 +02:00
return NT_STATUS_UNSUCCESSFUL ;
2008-01-16 12:09:48 +03:00
}
2008-08-05 22:38:44 +02:00
if ( rec - > value . dptr = = NULL ) {
2009-07-29 14:32:31 +02:00
val = * ( state - > oldval ) ;
2008-08-05 22:38:44 +02:00
} else if ( rec - > value . dsize = = sizeof ( val ) ) {
2008-01-16 12:09:48 +03:00
val = IVAL ( rec - > value . dptr , 0 ) ;
2009-07-29 14:32:31 +02:00
* ( state - > oldval ) = val ;
2008-08-05 22:38:44 +02:00
} else {
2009-07-29 14:22:37 +02:00
ret = NT_STATUS_UNSUCCESSFUL ;
goto done ;
2008-01-16 12:09:48 +03:00
}
2009-07-29 14:32:31 +02:00
val + = state - > change_val ;
2008-01-16 12:09:48 +03:00
2009-11-21 00:08:38 +01:00
SIVAL ( & v_store , 0 , val ) ;
2008-01-16 12:09:48 +03:00
2009-11-21 00:08:38 +01:00
ret = rec - > store ( rec ,
make_tdb_data ( ( const uint8 * ) & v_store ,
sizeof ( v_store ) ) ,
TDB_REPLACE ) ;
2008-01-16 12:09:48 +03:00
2009-07-29 14:22:37 +02:00
done :
2008-01-16 12:09:48 +03:00
TALLOC_FREE ( rec ) ;
2009-07-29 14:16:11 +02:00
return ret ;
2008-01-16 12:09:48 +03:00
}
2009-07-29 14:32:31 +02:00
NTSTATUS dbwrap_change_uint32_atomic ( struct db_context * db , const char * keystr ,
uint32_t * oldval , uint32_t change_val )
{
NTSTATUS ret ;
struct dbwrap_change_uint32_atomic_context state ;
state . keystr = keystr ;
state . oldval = oldval ;
state . change_val = change_val ;
ret = dbwrap_change_uint32_atomic_action ( db , & state ) ;
return ret ;
}
2009-07-29 14:37:39 +02:00
NTSTATUS dbwrap_trans_change_uint32_atomic ( struct db_context * db ,
const char * keystr ,
uint32_t * oldval ,
uint32_t change_val )
{
NTSTATUS ret ;
struct dbwrap_change_uint32_atomic_context state ;
state . keystr = keystr ;
state . oldval = oldval ;
state . change_val = change_val ;
ret = dbwrap_trans_do ( db , dbwrap_change_uint32_atomic_action , & state ) ;
return ret ;
}
2008-08-05 23:38:56 +02:00
/**
* Atomic integer change ( addition ) :
*
* if value does not exist yet in the db , use * oldval as initial old value .
* return old value in * oldval .
* store * oldval + change_val to db .
*/
2009-07-29 15:12:39 +02:00
struct dbwrap_change_int32_atomic_context {
const char * keystr ;
int32_t * oldval ;
int32_t change_val ;
} ;
static NTSTATUS dbwrap_change_int32_atomic_action ( struct db_context * db ,
void * private_data )
2008-03-09 11:15:10 +01:00
{
struct db_record * rec ;
2009-07-29 15:08:06 +02:00
int32_t val = - 1 ;
2009-11-21 00:14:29 +01:00
int32_t v_store ;
2009-07-29 14:21:03 +02:00
NTSTATUS ret ;
2009-07-29 15:12:39 +02:00
struct dbwrap_change_int32_atomic_context * state ;
2008-03-09 11:15:10 +01:00
2009-07-29 15:12:39 +02:00
state = ( struct dbwrap_change_int32_atomic_context * ) private_data ;
rec = db - > fetch_locked ( db , NULL , string_term_tdb_data ( state - > keystr ) ) ;
2009-07-29 14:26:26 +02:00
if ( ! rec ) {
2009-07-29 14:21:03 +02:00
return NT_STATUS_UNSUCCESSFUL ;
2008-03-09 11:15:10 +01:00
}
2008-08-05 23:14:05 +02:00
if ( rec - > value . dptr = = NULL ) {
2009-07-29 15:12:39 +02:00
val = * ( state - > oldval ) ;
2008-08-05 23:14:05 +02:00
} else if ( rec - > value . dsize = = sizeof ( val ) ) {
2008-03-09 11:15:10 +01:00
val = IVAL ( rec - > value . dptr , 0 ) ;
2009-07-29 15:12:39 +02:00
* ( state - > oldval ) = val ;
2008-08-05 23:14:05 +02:00
} else {
2009-07-29 14:23:26 +02:00
ret = NT_STATUS_UNSUCCESSFUL ;
goto done ;
2008-03-09 11:15:10 +01:00
}
2009-07-29 15:12:39 +02:00
val + = state - > change_val ;
2008-03-09 11:15:10 +01:00
2009-11-21 00:14:29 +01:00
SIVAL ( & v_store , 0 , val ) ;
2008-03-09 11:15:10 +01:00
2009-11-21 00:14:29 +01:00
ret = rec - > store ( rec ,
make_tdb_data ( ( const uint8_t * ) & v_store ,
sizeof ( v_store ) ) ,
TDB_REPLACE ) ;
2008-03-09 11:15:10 +01:00
2009-07-29 14:23:26 +02:00
done :
2008-03-09 11:15:10 +01:00
TALLOC_FREE ( rec ) ;
2009-07-29 14:21:03 +02:00
return ret ;
2008-03-09 11:15:10 +01:00
}
2008-03-10 13:27:27 +01:00
2009-07-29 15:12:39 +02:00
NTSTATUS dbwrap_change_int32_atomic ( struct db_context * db , const char * keystr ,
int32_t * oldval , int32_t change_val )
{
NTSTATUS ret ;
struct dbwrap_change_int32_atomic_context state ;
state . keystr = keystr ;
state . oldval = oldval ;
state . change_val = change_val ;
ret = dbwrap_change_int32_atomic_action ( db , & state ) ;
return ret ;
}
2009-07-29 15:14:57 +02:00
NTSTATUS dbwrap_trans_change_int32_atomic ( struct db_context * db ,
const char * keystr ,
int32_t * oldval ,
int32_t change_val )
{
NTSTATUS ret ;
struct dbwrap_change_int32_atomic_context state ;
state . keystr = keystr ;
state . oldval = oldval ;
state . change_val = change_val ;
ret = dbwrap_trans_do ( db , dbwrap_change_int32_atomic_action , & state ) ;
return ret ;
}
2009-07-21 12:35:48 +02:00
struct dbwrap_store_context {
TDB_DATA * key ;
TDB_DATA * dbuf ;
int flag ;
} ;
static NTSTATUS dbwrap_store_action ( struct db_context * db , void * private_data )
2008-03-10 13:27:27 +01:00
{
2008-03-30 13:11:22 +02:00
struct db_record * rec = NULL ;
2008-03-10 13:27:27 +01:00
NTSTATUS status ;
2009-07-21 12:35:48 +02:00
struct dbwrap_store_context * store_ctx ;
2008-03-10 13:27:27 +01:00
2009-07-21 12:35:48 +02:00
store_ctx = ( struct dbwrap_store_context * ) private_data ;
2008-03-10 13:27:27 +01:00
2009-07-21 12:35:48 +02:00
rec = db - > fetch_locked ( db , talloc_tos ( ) , * ( store_ctx - > key ) ) ;
2008-03-10 13:27:27 +01:00
if ( rec = = NULL ) {
DEBUG ( 5 , ( " fetch_locked failed \n " ) ) ;
2009-07-21 12:35:48 +02:00
return NT_STATUS_NO_MEMORY ;
2008-03-10 13:27:27 +01:00
}
2009-07-21 12:35:48 +02:00
status = rec - > store ( rec , * ( store_ctx - > dbuf ) , store_ctx - > flag ) ;
2008-03-10 13:27:27 +01:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 5 , ( " store returned %s \n " , nt_errstr ( status ) ) ) ;
}
TALLOC_FREE ( rec ) ;
2009-07-21 12:35:48 +02:00
return status ;
}
2008-03-10 13:27:27 +01:00
2009-07-21 12:35:48 +02:00
NTSTATUS dbwrap_trans_store ( struct db_context * db , TDB_DATA key , TDB_DATA dbuf ,
int flag )
{
NTSTATUS status ;
struct dbwrap_store_context store_ctx ;
2008-03-10 13:27:27 +01:00
2009-07-21 12:35:48 +02:00
store_ctx . key = & key ;
store_ctx . dbuf = & dbuf ;
store_ctx . flag = flag ;
2008-03-10 13:27:27 +01:00
2009-07-21 12:35:48 +02:00
status = dbwrap_trans_do ( db , dbwrap_store_action , & store_ctx ) ;
2008-03-30 13:11:22 +02:00
2008-03-28 11:53:00 +01:00
return status ;
2008-03-10 13:27:27 +01:00
}
2009-07-21 12:26:14 +02:00
static NTSTATUS dbwrap_delete_action ( struct db_context * db , void * private_data )
2008-03-10 13:27:27 +01:00
{
NTSTATUS status ;
2009-07-21 12:26:14 +02:00
struct db_record * rec ;
TDB_DATA * key = ( TDB_DATA * ) private_data ;
2008-03-10 13:27:27 +01:00
2009-07-21 12:26:14 +02:00
rec = db - > fetch_locked ( db , talloc_tos ( ) , * key ) ;
2008-03-10 13:27:27 +01:00
if ( rec = = NULL ) {
DEBUG ( 5 , ( " fetch_locked failed \n " ) ) ;
2009-07-21 12:26:14 +02:00
return NT_STATUS_NO_MEMORY ;
2008-03-10 13:27:27 +01:00
}
status = rec - > delete_rec ( rec ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 5 , ( " delete_rec returned %s \n " , nt_errstr ( status ) ) ) ;
}
2009-07-21 12:26:14 +02:00
talloc_free ( rec ) ;
return status ;
}
2008-03-10 13:27:27 +01:00
2009-07-21 12:26:14 +02:00
NTSTATUS dbwrap_trans_delete ( struct db_context * db , TDB_DATA key )
{
NTSTATUS status ;
2008-03-10 13:27:27 +01:00
2009-07-21 12:26:14 +02:00
status = dbwrap_trans_do ( db , dbwrap_delete_action , & key ) ;
2008-04-07 23:36:29 +02:00
2008-03-28 11:57:54 +01:00
return status ;
2008-03-10 13:27:27 +01:00
}
2008-03-27 15:59:58 +01:00
2008-03-28 11:53:00 +01:00
NTSTATUS dbwrap_trans_store_int32 ( struct db_context * db , const char * keystr ,
int32_t v )
2008-03-27 15:59:58 +01:00
{
int32 v_store ;
SIVAL ( & v_store , 0 , v ) ;
2008-03-28 11:53:00 +01:00
return dbwrap_trans_store ( db , string_term_tdb_data ( keystr ) ,
make_tdb_data ( ( const uint8 * ) & v_store ,
2008-03-27 15:59:58 +01:00
sizeof ( v_store ) ) ,
2008-03-28 11:53:00 +01:00
TDB_REPLACE ) ;
2008-03-27 15:59:58 +01:00
}
2008-03-28 12:02:28 +01:00
2008-03-28 20:03:30 +01:00
NTSTATUS dbwrap_trans_store_uint32 ( struct db_context * db , const char * keystr ,
uint32_t v )
{
uint32 v_store ;
SIVAL ( & v_store , 0 , v ) ;
return dbwrap_trans_store ( db , string_term_tdb_data ( keystr ) ,
make_tdb_data ( ( const uint8 * ) & v_store ,
sizeof ( v_store ) ) ,
TDB_REPLACE ) ;
}
2008-03-28 12:02:28 +01:00
NTSTATUS dbwrap_trans_store_bystring ( struct db_context * db , const char * key ,
TDB_DATA data , int flags )
{
return dbwrap_trans_store ( db , string_term_tdb_data ( key ) , data , flags ) ;
}
NTSTATUS dbwrap_trans_delete_bystring ( struct db_context * db , const char * key )
{
return dbwrap_trans_delete ( db , string_term_tdb_data ( key ) ) ;
}
2009-07-15 14:00:42 +02:00
/**
* Wrap db action ( s ) into a transaction .
*/
NTSTATUS dbwrap_trans_do ( struct db_context * db ,
NTSTATUS ( * action ) ( struct db_context * , void * ) ,
void * private_data )
{
int res ;
NTSTATUS status ;
res = db - > transaction_start ( db ) ;
if ( res ! = 0 ) {
DEBUG ( 5 , ( " transaction_start failed \n " ) ) ;
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
status = action ( db , private_data ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
if ( db - > transaction_cancel ( db ) ! = 0 ) {
smb_panic ( " Cancelling transaction failed " ) ;
}
return status ;
}
res = db - > transaction_commit ( db ) ;
if ( res = = 0 ) {
return NT_STATUS_OK ;
}
DEBUG ( 2 , ( " transaction_commit failed \n " ) ) ;
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
2009-07-08 16:02:19 +02:00
2011-03-16 09:13:40 +01:00
struct dbwrap_trans_traverse_action_ctx {
int ( * f ) ( struct db_record * rec , void * private_data ) ;
void * private_data ;
} ;
static NTSTATUS dbwrap_trans_traverse_action ( struct db_context * db , void * private_data )
{
struct dbwrap_trans_traverse_action_ctx * ctx =
( struct dbwrap_trans_traverse_action_ctx * ) private_data ;
int ret = db - > traverse ( db , ctx - > f , ctx - > private_data ) ;
return ( ret = = - 1 ) ? NT_STATUS_INTERNAL_DB_CORRUPTION : NT_STATUS_OK ;
}
NTSTATUS dbwrap_trans_traverse ( struct db_context * db ,
int ( * f ) ( struct db_record * , void * ) ,
void * private_data )
{
struct dbwrap_trans_traverse_action_ctx ctx = {
. f = f ,
. private_data = private_data ,
} ;
return dbwrap_trans_do ( db , dbwrap_trans_traverse_action , & ctx ) ;
}
2011-03-18 14:39:15 +01:00
NTSTATUS dbwrap_traverse ( struct db_context * db ,
int ( * f ) ( struct db_record * , void * ) ,
void * private_data )
{
int ret = db - > traverse ( db , f , private_data ) ;
return ( ret = = - 1 ) ? NT_STATUS_INTERNAL_DB_CORRUPTION : NT_STATUS_OK ;
}
2011-03-16 09:13:40 +01:00
2011-05-30 17:27:23 +10:00
NTSTATUS dbwrap_delete ( struct db_context * db , TDB_DATA key )
{
struct db_record * rec ;
NTSTATUS status ;
rec = db - > fetch_locked ( db , talloc_tos ( ) , key ) ;
if ( rec = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
status = rec - > delete_rec ( rec ) ;
TALLOC_FREE ( rec ) ;
return status ;
}
NTSTATUS dbwrap_store ( struct db_context * db , TDB_DATA key ,
TDB_DATA data , int flags )
{
struct db_record * rec ;
NTSTATUS status ;
rec = db - > fetch_locked ( db , talloc_tos ( ) , key ) ;
if ( rec = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
status = rec - > store ( rec , data , flags ) ;
TALLOC_FREE ( rec ) ;
return status ;
}
TDB_DATA dbwrap_fetch ( struct db_context * db , TALLOC_CTX * mem_ctx ,
TDB_DATA key )
{
TDB_DATA result ;
if ( db - > fetch ( db , mem_ctx , key , & result ) = = - 1 ) {
return make_tdb_data ( NULL , 0 ) ;
}
return result ;
}
NTSTATUS dbwrap_delete_bystring ( struct db_context * db , const char * key )
{
return dbwrap_delete ( db , string_term_tdb_data ( key ) ) ;
}
NTSTATUS dbwrap_store_bystring ( struct db_context * db , const char * key ,
TDB_DATA data , int flags )
{
return dbwrap_store ( db , string_term_tdb_data ( key ) , data , flags ) ;
}
TDB_DATA dbwrap_fetch_bystring ( struct db_context * db , TALLOC_CTX * mem_ctx ,
const char * key )
{
return dbwrap_fetch ( db , mem_ctx , string_term_tdb_data ( key ) ) ;
}
2011-03-16 09:13:40 +01:00
2009-07-08 16:02:19 +02:00
NTSTATUS dbwrap_delete_bystring_upper ( struct db_context * db , const char * key )
{
char * key_upper ;
NTSTATUS status ;
key_upper = talloc_strdup_upper ( talloc_tos ( ) , key ) ;
if ( key_upper = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
status = dbwrap_delete_bystring ( db , key_upper ) ;
talloc_free ( key_upper ) ;
return status ;
}
2009-07-08 16:08:41 +02:00
NTSTATUS dbwrap_store_bystring_upper ( struct db_context * db , const char * key ,
TDB_DATA data , int flags )
{
char * key_upper ;
NTSTATUS status ;
key_upper = talloc_strdup_upper ( talloc_tos ( ) , key ) ;
if ( key_upper = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
status = dbwrap_store_bystring ( db , key_upper , data , flags ) ;
talloc_free ( key_upper ) ;
return status ;
}
2009-07-08 16:13:07 +02:00
TDB_DATA dbwrap_fetch_bystring_upper ( struct db_context * db , TALLOC_CTX * mem_ctx ,
const char * key )
{
char * key_upper ;
TDB_DATA result ;
key_upper = talloc_strdup_upper ( talloc_tos ( ) , key ) ;
if ( key_upper = = NULL ) {
return make_tdb_data ( NULL , 0 ) ;
}
result = dbwrap_fetch_bystring ( db , mem_ctx , key_upper ) ;
talloc_free ( key_upper ) ;
return result ;
}