2008-01-16 12:09:48 +03:00
/*
Unix SMB / CIFS implementation .
Utility functions for the dbwrap API
Copyright ( C ) Volker Lendecke 2007
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"
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 .
*/
2008-01-16 12:09:48 +03:00
uint32_t dbwrap_change_uint32_atomic ( struct db_context * db , const char * keystr ,
uint32_t * oldval , uint32_t change_val )
{
struct db_record * rec ;
uint32 val = - 1 ;
TDB_DATA data ;
if ( ! ( rec = db - > fetch_locked ( db , NULL ,
string_term_tdb_data ( keystr ) ) ) ) {
return - 1 ;
}
2008-08-05 22:38:44 +02:00
if ( rec - > value . dptr = = NULL ) {
val = * oldval ;
} else if ( rec - > value . dsize = = sizeof ( val ) ) {
2008-01-16 12:09:48 +03:00
val = IVAL ( rec - > value . dptr , 0 ) ;
2008-08-05 22:38:44 +02:00
* oldval = val ;
} else {
return - 1 ;
2008-01-16 12:09:48 +03:00
}
val + = change_val ;
data . dsize = sizeof ( val ) ;
data . dptr = ( uint8 * ) & val ;
rec - > store ( rec , data , TDB_REPLACE ) ;
TALLOC_FREE ( rec ) ;
return 0 ;
}
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 .
*/
2008-03-09 11:15:10 +01:00
int32 dbwrap_change_int32_atomic ( struct db_context * db , const char * keystr ,
int32 * oldval , int32 change_val )
{
struct db_record * rec ;
int32 val = - 1 ;
TDB_DATA data ;
if ( ! ( rec = db - > fetch_locked ( db , NULL ,
string_term_tdb_data ( keystr ) ) ) ) {
return - 1 ;
}
2008-08-05 23:14:05 +02:00
if ( rec - > value . dptr = = NULL ) {
val = * oldval ;
} else if ( rec - > value . dsize = = sizeof ( val ) ) {
2008-03-09 11:15:10 +01:00
val = IVAL ( rec - > value . dptr , 0 ) ;
2008-08-05 23:14:05 +02:00
* oldval = val ;
} else {
return - 1 ;
2008-03-09 11:15:10 +01:00
}
val + = change_val ;
data . dsize = sizeof ( val ) ;
data . dptr = ( uint8 * ) & val ;
rec - > store ( rec , data , TDB_REPLACE ) ;
TALLOC_FREE ( rec ) ;
return 0 ;
}
2008-03-10 13:27:27 +01:00
2008-03-28 11:53:00 +01:00
NTSTATUS dbwrap_trans_store ( struct db_context * db , TDB_DATA key , TDB_DATA dbuf ,
int flag )
2008-03-10 13:27:27 +01:00
{
int res ;
2008-03-30 13:11:22 +02:00
struct db_record * rec = NULL ;
2008-03-10 13:27:27 +01:00
NTSTATUS status ;
res = db - > transaction_start ( db ) ;
if ( res ! = 0 ) {
DEBUG ( 5 , ( " transaction_start failed \n " ) ) ;
2008-03-28 11:53:00 +01:00
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
2008-03-10 13:27:27 +01:00
}
rec = db - > fetch_locked ( db , talloc_tos ( ) , key ) ;
if ( rec = = NULL ) {
DEBUG ( 5 , ( " fetch_locked failed \n " ) ) ;
2008-03-28 11:53:00 +01:00
status = NT_STATUS_NO_MEMORY ;
2008-03-10 13:27:27 +01:00
goto cancel ;
}
status = rec - > store ( rec , dbuf , flag ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 5 , ( " store returned %s \n " , nt_errstr ( status ) ) ) ;
goto cancel ;
}
TALLOC_FREE ( rec ) ;
res = db - > transaction_commit ( db ) ;
if ( res ! = 0 ) {
DEBUG ( 5 , ( " tdb_transaction_commit failed \n " ) ) ;
2008-03-28 11:53:00 +01:00
status = NT_STATUS_INTERNAL_DB_CORRUPTION ;
2008-08-08 11:41:21 +10:00
TALLOC_FREE ( rec ) ;
return status ;
2008-03-10 13:27:27 +01:00
}
2008-03-28 11:53:00 +01:00
return NT_STATUS_OK ;
2008-03-10 13:27:27 +01:00
cancel :
2008-03-30 13:11:22 +02:00
TALLOC_FREE ( rec ) ;
2008-03-10 13:27:27 +01:00
if ( db - > transaction_cancel ( db ) ! = 0 ) {
smb_panic ( " Cancelling transaction failed " ) ;
}
2008-03-28 11:53:00 +01:00
return status ;
2008-03-10 13:27:27 +01:00
}
2008-03-28 11:57:54 +01:00
NTSTATUS dbwrap_trans_delete ( struct db_context * db , TDB_DATA key )
2008-03-10 13:27:27 +01:00
{
int res ;
2008-04-07 23:36:29 +02:00
struct db_record * rec = NULL ;
2008-03-10 13:27:27 +01:00
NTSTATUS status ;
res = db - > transaction_start ( db ) ;
if ( res ! = 0 ) {
DEBUG ( 5 , ( " transaction_start failed \n " ) ) ;
2008-03-28 11:57:54 +01:00
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
2008-03-10 13:27:27 +01:00
}
rec = db - > fetch_locked ( db , talloc_tos ( ) , key ) ;
if ( rec = = NULL ) {
DEBUG ( 5 , ( " fetch_locked failed \n " ) ) ;
2008-03-28 11:57:54 +01:00
status = NT_STATUS_NO_MEMORY ;
2008-03-10 13:27:27 +01:00
goto cancel ;
}
status = rec - > delete_rec ( rec ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 5 , ( " delete_rec returned %s \n " , nt_errstr ( status ) ) ) ;
goto cancel ;
}
TALLOC_FREE ( rec ) ;
res = db - > transaction_commit ( db ) ;
if ( res ! = 0 ) {
DEBUG ( 5 , ( " tdb_transaction_commit failed \n " ) ) ;
2008-03-28 11:57:54 +01:00
status = NT_STATUS_INTERNAL_DB_CORRUPTION ;
2008-08-08 11:41:21 +10:00
TALLOC_FREE ( rec ) ;
return status ;
2008-03-10 13:27:27 +01:00
}
2008-03-28 11:57:54 +01:00
return NT_STATUS_OK ;
2008-03-10 13:27:27 +01:00
cancel :
2008-04-07 23:36:29 +02:00
TALLOC_FREE ( rec ) ;
2008-03-10 13:27:27 +01:00
if ( db - > transaction_cancel ( db ) ! = 0 ) {
smb_panic ( " Cancelling transaction failed " ) ;
}
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 ) ) ;
}