2013-12-15 20:23:16 +01:00
/*
2011-05-04 10:28:15 +10:00
Unix SMB / CIFS implementation .
TDB wrap functions
Copyright ( C ) Andrew Tridgell 2004
Copyright ( C ) Jelmer Vernooij < jelmer @ samba . org > 2007
2013-12-15 20:23:16 +01:00
2011-05-04 10:28:15 +10: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 .
2013-12-15 20:23:16 +01:00
2011-05-04 10:28:15 +10:00
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 .
2013-12-15 20:23:16 +01:00
2011-05-04 10:28:15 +10:00
You should have received a copy of the GNU General Public License
along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
2014-08-19 12:22:00 +10:00
# include "replace.h"
2011-05-04 10:28:15 +10:00
# include "lib/util/dlinklist.h"
2014-08-19 12:22:00 +10:00
# include "lib/util/debug.h"
# include "tdb_wrap.h"
2011-05-04 10:28:15 +10:00
/*
Log tdb messages via DEBUG ( ) .
*/
2013-12-15 20:23:16 +01:00
static void tdb_wrap_log ( TDB_CONTEXT * tdb , enum tdb_debug_level level ,
2011-05-04 10:28:15 +10:00
const char * format , . . . ) PRINTF_ATTRIBUTE ( 3 , 4 ) ;
2013-12-15 20:23:16 +01:00
static void tdb_wrap_log ( TDB_CONTEXT * tdb , enum tdb_debug_level level ,
2011-05-04 10:28:15 +10:00
const char * format , . . . )
{
va_list ap ;
char * ptr = NULL ;
int debuglevel = 0 ;
int ret ;
switch ( level ) {
case TDB_DEBUG_FATAL :
debuglevel = 0 ;
break ;
case TDB_DEBUG_ERROR :
debuglevel = 1 ;
break ;
case TDB_DEBUG_WARNING :
debuglevel = 2 ;
break ;
case TDB_DEBUG_TRACE :
debuglevel = 5 ;
break ;
default :
debuglevel = 0 ;
2013-12-15 20:23:16 +01:00
}
2011-05-04 10:28:15 +10:00
va_start ( ap , format ) ;
ret = vasprintf ( & ptr , format , ap ) ;
va_end ( ap ) ;
if ( ret ! = - 1 ) {
const char * name = tdb_name ( tdb ) ;
DEBUG ( debuglevel , ( " tdb(%s): %s " , name ? name : " unnamed " , ptr ) ) ;
free ( ptr ) ;
}
}
struct tdb_wrap_private {
struct tdb_context * tdb ;
const char * name ;
struct tdb_wrap_private * next , * prev ;
} ;
static struct tdb_wrap_private * tdb_list ;
/* destroy the last connection to a tdb */
static int tdb_wrap_private_destructor ( struct tdb_wrap_private * w )
{
tdb_close ( w - > tdb ) ;
DLIST_REMOVE ( tdb_list , w ) ;
return 0 ;
2013-12-15 20:23:16 +01:00
}
2011-05-04 10:28:15 +10:00
static struct tdb_wrap_private * tdb_wrap_private_open ( TALLOC_CTX * mem_ctx ,
const char * name ,
int hash_size ,
int tdb_flags ,
int open_flags ,
2014-03-26 13:32:16 +00:00
mode_t mode )
2011-05-04 10:28:15 +10:00
{
struct tdb_wrap_private * result ;
2015-07-12 12:45:33 +02:00
struct tdb_logging_context lctx = { . log_fn = tdb_wrap_log } ;
2011-05-04 10:28:15 +10:00
2015-07-12 12:37:43 +02:00
result = talloc_pooled_object ( mem_ctx , struct tdb_wrap_private ,
1 , strlen ( name ) + 1 ) ;
2011-05-04 10:28:15 +10:00
if ( result = = NULL ) {
return NULL ;
}
2015-07-12 12:37:43 +02:00
/* Doesn't fail, see talloc_pooled_object */
2011-05-04 10:28:15 +10:00
result - > name = talloc_strdup ( result , name ) ;
2012-06-19 12:43:10 +09:30
result - > tdb = tdb_open_ex ( name , hash_size , tdb_flags ,
open_flags , mode , & lctx , NULL ) ;
2011-05-04 10:28:15 +10:00
if ( result - > tdb = = NULL ) {
goto fail ;
}
talloc_set_destructor ( result , tdb_wrap_private_destructor ) ;
DLIST_ADD ( tdb_list , result ) ;
return result ;
fail :
TALLOC_FREE ( result ) ;
return NULL ;
}
/*
wrapped connection to a tdb database
to close just talloc_free ( ) the tdb_wrap pointer
*/
2014-03-26 14:41:03 +00:00
struct tdb_wrap * tdb_wrap_open ( TALLOC_CTX * mem_ctx ,
const char * name , int hash_size , int tdb_flags ,
int open_flags , mode_t mode )
2011-05-04 10:28:15 +10:00
{
struct tdb_wrap * result ;
struct tdb_wrap_private * w ;
2015-01-19 12:37:13 +01:00
if ( name = = NULL ) {
errno = EINVAL ;
return NULL ;
}
2011-05-04 10:28:15 +10:00
result = talloc ( mem_ctx , struct tdb_wrap ) ;
if ( result = = NULL ) {
return NULL ;
}
for ( w = tdb_list ; w ; w = w - > next ) {
if ( strcmp ( name , w - > name ) = = 0 ) {
break ;
}
}
if ( w = = NULL ) {
2014-09-30 11:04:21 +00:00
if ( tdb_flags & TDB_MUTEX_LOCKING ) {
if ( ! tdb_runtime_check_for_robust_mutexes ( ) ) {
tdb_flags & = ~ TDB_MUTEX_LOCKING ;
}
}
2011-05-04 10:28:15 +10:00
w = tdb_wrap_private_open ( result , name , hash_size , tdb_flags ,
2014-03-26 13:32:16 +00:00
open_flags , mode ) ;
2011-05-04 10:28:15 +10:00
} else {
/*
* Correctly use talloc_reference : The tdb will be
* closed when " w " is being freed . The caller never
* sees " w " , so an incorrect use of talloc_free ( w )
* instead of calling talloc_unlink is not possible .
* To avoid having to refcount ourselves , " w " will
* have multiple parents that hang off all the
* tdb_wrap ' s being returned from here . Those parents
* can be freed without problem .
*/
if ( talloc_reference ( result , w ) = = NULL ) {
goto fail ;
}
}
if ( w = = NULL ) {
goto fail ;
}
result - > tdb = w - > tdb ;
return result ;
fail :
TALLOC_FREE ( result ) ;
return NULL ;
}