0001-01-01 02:30:17 +02:30
/*
* Unix SMB / CIFS implementation .
2005-05-23 20:25:31 +04:00
* Virtual Windows Registry Layer
* Copyright ( C ) Gerald Carter 2002 - 2005
0001-01-01 02:30:17 +02:30
*
* 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
2007-07-09 23:25:36 +04:00
* the Free Software Foundation ; either version 3 of the License , or
0001-01-01 02:30:17 +02:30
* ( 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
2007-07-10 09:23:25 +04:00
* along with this program ; if not , see < http : //www.gnu.org/licenses/>.
0001-01-01 02:30:17 +02:30
*/
/* Implementation of internal registry database functions. */
# include "includes.h"
# undef DBGC_CLASS
2007-09-29 03:05:52 +04:00
# define DBGC_CLASS DBGC_REGISTRY
0001-01-01 02:30:17 +02:30
2007-06-15 14:40:36 +04:00
static struct tdb_wrap * tdb_reg = NULL ;
2005-10-07 16:14:25 +04:00
static int tdb_refcount ;
0001-01-01 02:30:17 +02:30
2005-05-23 20:25:31 +04:00
/* List the deepest path into the registry. All part components will be created.*/
2005-08-29 18:55:40 +04:00
/* If you want to have a part of the path controlled by the tdb and part by
2005-05-23 20:25:31 +04:00
a virtual registry db ( e . g . printing ) , then you have to list the deepest path .
For example , " HKLM/SOFTWARE/Microsoft/Windows NT/CurrentVersion/Print "
allows the reg_db backend to handle everything up to
" HKLM/SOFTWARE/Microsoft/Windows NT/CurrentVersion " and then we ' ll hook
the reg_printing backend onto the last component of the path ( see
KEY_PRINTING_2K in include / rpc_reg . h ) - - jerry */
static const char * builtin_registry_paths [ ] = {
2005-06-30 23:43:53 +04:00
KEY_PRINTING_2K ,
KEY_PRINTING_PORTS ,
KEY_PRINTING ,
KEY_SHARES ,
KEY_EVENTLOG ,
2006-11-30 10:38:40 +03:00
KEY_SMBCONF ,
2008-01-20 05:39:27 +03:00
KEY_PERFLIB ,
KEY_PERFLIB_009 ,
2005-07-02 02:24:00 +04:00
" HKLM \\ SYSTEM \\ CurrentControlSet \\ Control \\ Print \\ Monitors " ,
2008-01-18 18:15:43 +03:00
KEY_PROD_OPTIONS ,
2005-09-01 18:58:57 +04:00
" HKLM \\ SYSTEM \\ CurrentControlSet \\ Control \\ Terminal Server \\ DefaultUserConfiguration " ,
2008-01-20 01:46:13 +03:00
KEY_TCPIP_PARAMS ,
2008-01-18 18:13:01 +03:00
KEY_NETLOGON_PARAMS ,
2008-01-20 03:09:52 +03:00
KEY_HKU ,
KEY_HKCR ,
KEY_HKPD ,
KEY_HKPT ,
2005-05-23 20:25:31 +04:00
NULL } ;
2005-06-17 19:53:01 +04:00
2005-06-25 02:34:40 +04:00
struct builtin_regkey_value {
const char * path ;
const char * valuename ;
uint32 type ;
union {
const char * string ;
uint32 dw_value ;
} data ;
} ;
2005-06-25 21:31:40 +04:00
static struct builtin_regkey_value builtin_registry_values [ ] = {
2005-08-05 18:34:25 +04:00
{ KEY_PRINTING_PORTS ,
2005-07-04 20:52:29 +04:00
SAMBA_PRINTER_PORT_NAME , REG_SZ , { " " } } ,
2005-08-05 18:34:25 +04:00
{ KEY_PRINTING_2K ,
2005-07-04 20:52:29 +04:00
" DefaultSpoolDirectory " , REG_SZ , { " C: \\ Windows \\ System32 \\ Spool \\ Printers " } } ,
2005-08-05 18:34:25 +04:00
{ KEY_EVENTLOG ,
" DisplayName " , REG_SZ , { " Event Log " } } ,
{ KEY_EVENTLOG ,
" ErrorControl " , REG_DWORD , { ( char * ) 0x00000001 } } ,
2005-06-25 21:31:40 +04:00
{ NULL , NULL , 0 , { NULL } }
2005-06-25 02:34:40 +04:00
} ;
0001-01-01 02:30:17 +02:30
/***********************************************************************
Open the registry data in the tdb
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-11-27 04:24:56 +03:00
2007-10-19 04:40:25 +04:00
static bool init_registry_data ( void )
0001-01-01 02:30:17 +02:30
{
2007-11-27 04:24:56 +03:00
char * path = NULL ;
char * base = NULL ;
char * remaining = NULL ;
2007-12-14 00:20:58 +03:00
TALLOC_CTX * frame = NULL ;
2007-12-08 04:32:32 +03:00
char * keyname ;
char * subkeyname ;
2006-12-03 20:34:11 +03:00
REGSUBKEY_CTR * subkeys ;
REGVAL_CTR * values ;
2005-05-23 20:25:31 +04:00
int i ;
2006-12-03 20:34:11 +03:00
const char * p , * p2 ;
UNISTR2 data ;
2007-02-07 16:26:13 +03:00
/*
* There are potentially quite a few store operations which are all
* indiviually wrapped in tdb transactions . Wrapping them in a single
* transaction gives just a single transaction_commit ( ) to actually do
* its fsync ( ) s . See tdb / common / transaction . c for info about nested
* transaction behaviour .
*/
2007-06-15 14:40:36 +04:00
if ( tdb_transaction_start ( tdb_reg - > tdb ) = = - 1 ) {
2007-02-07 16:26:13 +03:00
DEBUG ( 0 , ( " init_registry_data: tdb_transaction_start "
" failed \n " ) ) ;
2007-11-27 04:24:56 +03:00
return false ;
2007-02-07 16:26:13 +03:00
}
2007-11-27 04:24:56 +03:00
2005-05-23 20:25:31 +04:00
/* loop over all of the predefined paths and add each component */
2007-11-27 04:24:56 +03:00
2005-05-23 20:25:31 +04:00
for ( i = 0 ; builtin_registry_paths [ i ] ! = NULL ; i + + ) {
2006-12-03 20:34:11 +03:00
2007-12-14 00:20:58 +03:00
frame = talloc_stackframe ( ) ;
2006-12-03 20:34:11 +03:00
DEBUG ( 6 , ( " init_registry_data: Adding [%s] \n " , builtin_registry_paths [ i ] ) ) ;
2007-12-14 00:20:58 +03:00
path = talloc_strdup ( talloc_tos ( ) , builtin_registry_paths [ i ] ) ;
base = talloc_strdup ( talloc_tos ( ) , " " ) ;
2007-11-27 04:24:56 +03:00
if ( ! path | | ! base ) {
goto fail ;
}
2006-12-03 20:34:11 +03:00
p = path ;
2007-11-27 04:24:56 +03:00
2007-12-14 00:20:58 +03:00
while ( next_token_talloc ( talloc_tos ( ) , & p , & keyname , " \\ " ) ) {
2007-11-27 04:24:56 +03:00
2006-12-03 20:34:11 +03:00
/* build up the registry path from the components */
2007-11-27 04:24:56 +03:00
if ( * base ) {
2007-12-14 00:20:58 +03:00
base = talloc_asprintf ( talloc_tos ( ) , " %s \\ " , base ) ;
2007-11-27 04:24:56 +03:00
if ( ! base ) {
goto fail ;
}
}
base = talloc_asprintf_append ( base , " %s " , keyname ) ;
if ( ! base ) {
goto fail ;
}
2006-12-03 20:34:11 +03:00
/* get the immediate subkeyname (if we have one ) */
2007-11-27 04:24:56 +03:00
2007-12-14 00:20:58 +03:00
subkeyname = talloc_strdup ( talloc_tos ( ) , " " ) ;
2007-12-08 04:32:32 +03:00
if ( ! subkeyname ) {
goto fail ;
}
2007-11-27 04:24:56 +03:00
if ( * p ) {
2007-12-14 00:20:58 +03:00
remaining = talloc_strdup ( talloc_tos ( ) , p ) ;
2007-11-27 04:24:56 +03:00
if ( ! remaining ) {
goto fail ;
}
2006-12-03 20:34:11 +03:00
p2 = remaining ;
2007-11-27 04:24:56 +03:00
2007-12-14 00:20:58 +03:00
if ( ! next_token_talloc ( talloc_tos ( ) , & p2 ,
2007-12-08 04:32:32 +03:00
& subkeyname , " \\ " ) ) {
2007-12-14 00:20:58 +03:00
subkeyname = talloc_strdup ( talloc_tos ( ) , p2 ) ;
2007-12-08 04:32:32 +03:00
if ( ! subkeyname ) {
goto fail ;
}
2007-11-27 04:24:56 +03:00
}
2006-12-03 20:34:11 +03:00
}
DEBUG ( 10 , ( " init_registry_data: Storing key [%s] with subkey [%s] \n " ,
base , * subkeyname ? subkeyname : " NULL " ) ) ;
2007-11-27 04:24:56 +03:00
2006-12-03 20:34:11 +03:00
/* we don't really care if the lookup succeeds or not since
2007-11-27 04:24:56 +03:00
we are about to update the record . We just want any
2006-12-03 20:34:11 +03:00
subkeys already present */
2007-11-27 04:24:56 +03:00
2007-12-14 00:20:58 +03:00
if ( ! ( subkeys = TALLOC_ZERO_P ( talloc_tos ( ) , REGSUBKEY_CTR ) ) ) {
2006-12-03 20:34:11 +03:00
DEBUG ( 0 , ( " talloc() failure! \n " ) ) ;
2007-02-07 16:26:13 +03:00
goto fail ;
2006-12-03 20:34:11 +03:00
}
2007-11-27 04:24:56 +03:00
regdb_fetch_keys ( base , subkeys ) ;
if ( * subkeyname ) {
regsubkey_ctr_addkey ( subkeys , subkeyname ) ;
}
if ( ! regdb_store_keys ( base , subkeys ) ) {
2007-02-07 16:26:13 +03:00
goto fail ;
2007-11-27 04:24:56 +03:00
}
2005-05-23 20:25:31 +04:00
}
2007-12-14 00:20:58 +03:00
TALLOC_FREE ( frame ) ;
2005-05-23 20:25:31 +04:00
}
2005-05-09 17:51:44 +04:00
2005-06-25 21:31:40 +04:00
/* loop over all of the predefined values and add each component */
2007-11-27 04:24:56 +03:00
for ( i = 0 ; builtin_registry_values [ i ] . path ! = NULL ; i + + ) {
2007-12-14 00:20:58 +03:00
if ( ! ( values = TALLOC_ZERO_P ( talloc_tos ( ) , REGVAL_CTR ) ) ) {
2007-02-07 16:26:13 +03:00
goto fail ;
2005-08-29 18:55:40 +04:00
}
2007-11-27 04:24:56 +03:00
regdb_fetch_values ( builtin_registry_values [ i ] . path , values ) ;
2006-12-03 20:34:11 +03:00
/* preserve existing values across restarts. Only add new ones */
2007-11-27 04:24:56 +03:00
if ( ! regval_ctr_key_exists ( values , builtin_registry_values [ i ] . valuename ) ) {
switch ( builtin_registry_values [ i ] . type ) {
2006-12-03 20:34:11 +03:00
case REG_DWORD :
2007-11-27 04:24:56 +03:00
regval_ctr_addvalue ( values ,
2006-12-03 20:34:11 +03:00
builtin_registry_values [ i ] . valuename ,
REG_DWORD ,
( char * ) & builtin_registry_values [ i ] . data . dw_value ,
sizeof ( uint32 ) ) ;
break ;
2007-11-27 04:24:56 +03:00
2006-12-03 20:34:11 +03:00
case REG_SZ :
init_unistr2 ( & data , builtin_registry_values [ i ] . data . string , UNI_STR_TERMINATE ) ;
2007-11-27 04:24:56 +03:00
regval_ctr_addvalue ( values ,
2006-12-03 20:34:11 +03:00
builtin_registry_values [ i ] . valuename ,
REG_SZ ,
( char * ) data . buffer ,
data . uni_str_len * sizeof ( uint16 ) ) ;
break ;
2007-11-27 04:24:56 +03:00
2006-12-03 20:34:11 +03:00
default :
DEBUG ( 0 , ( " init_registry_data: invalid value type in builtin_registry_values [%d] \n " ,
builtin_registry_values [ i ] . type ) ) ;
}
regdb_store_values ( builtin_registry_values [ i ] . path , values ) ;
2005-06-25 21:31:40 +04:00
}
2006-12-03 20:34:11 +03:00
TALLOC_FREE ( values ) ;
2005-06-25 21:31:40 +04:00
}
2007-11-27 04:24:56 +03:00
2007-12-14 00:20:58 +03:00
TALLOC_FREE ( frame ) ;
2007-06-15 14:40:36 +04:00
if ( tdb_transaction_commit ( tdb_reg - > tdb ) = = - 1 ) {
2007-02-07 16:26:13 +03:00
DEBUG ( 0 , ( " init_registry_data: Could not commit "
" transaction \n " ) ) ;
2007-11-27 04:24:56 +03:00
return false ;
2007-02-07 16:26:13 +03:00
}
2007-11-27 04:24:56 +03:00
return true ;
2007-02-07 16:26:13 +03:00
fail :
2007-12-14 00:20:58 +03:00
TALLOC_FREE ( frame ) ;
2007-06-15 14:40:36 +04:00
if ( tdb_transaction_cancel ( tdb_reg - > tdb ) = = - 1 ) {
2007-02-07 16:26:13 +03:00
smb_panic ( " init_registry_data: tdb_transaction_cancel "
" failed \n " ) ;
}
2007-11-27 04:24:56 +03:00
return false ;
0001-01-01 02:30:17 +02:30
}
/***********************************************************************
Open the registry database
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-19 04:40:25 +04:00
bool regdb_init ( void )
0001-01-01 02:30:17 +02:30
{
2005-06-17 19:53:01 +04:00
const char * vstring = " INFO/version " ;
uint32 vers_id ;
0001-01-01 02:30:17 +02:30
2008-02-19 03:04:31 +03:00
if ( tdb_reg ) {
DEBUG ( 10 , ( " regdb_init: incrementing refcount (%d) \n " , tdb_refcount ) ) ;
tdb_refcount + + ;
2008-01-04 17:20:46 +03:00
return true ;
2008-02-19 03:04:31 +03:00
}
0001-01-01 02:30:17 +02:30
2007-11-01 22:53:44 +03:00
if ( ! ( tdb_reg = tdb_wrap_open ( NULL , state_path ( " registry.tdb " ) , 0 , REG_TDB_FLAGS , O_RDWR , 0600 ) ) )
0001-01-01 02:30:17 +02:30
{
2007-11-01 22:53:44 +03:00
tdb_reg = tdb_wrap_open ( NULL , state_path ( " registry.tdb " ) , 0 , REG_TDB_FLAGS , O_RDWR | O_CREAT , 0600 ) ;
0001-01-01 02:30:17 +02:30
if ( ! tdb_reg ) {
2005-10-07 16:14:25 +04:00
DEBUG ( 0 , ( " regdb_init: Failed to open registry %s (%s) \n " ,
2007-11-01 22:53:44 +03:00
state_path ( " registry.tdb " ) , strerror ( errno ) ) ) ;
2008-01-04 17:20:46 +03:00
return false ;
0001-01-01 02:30:17 +02:30
}
2005-10-07 16:14:25 +04:00
DEBUG ( 10 , ( " regdb_init: Successfully created registry tdb \n " ) ) ;
2005-05-23 20:25:31 +04:00
}
2005-10-07 16:14:25 +04:00
tdb_refcount = 1 ;
2005-05-23 20:25:31 +04:00
2007-06-15 14:40:36 +04:00
vers_id = tdb_fetch_int32 ( tdb_reg - > tdb , vstring ) ;
2005-06-17 19:53:01 +04:00
if ( vers_id ! = REGVER_V1 ) {
2005-09-01 18:00:53 +04:00
/* any upgrade code here if needed */
2007-06-16 01:38:10 +04:00
DEBUG ( 10 , ( " regdb_init: got INFO/version = %d != %d \n " ,
vers_id , REGVER_V1 ) ) ;
2005-09-01 18:00:53 +04:00
}
2005-06-17 19:53:01 +04:00
2005-09-01 18:00:53 +04:00
/* always setup the necessary keys and values */
2005-06-17 19:53:01 +04:00
2005-09-01 18:00:53 +04:00
if ( ! init_registry_data ( ) ) {
2008-01-04 23:35:29 +03:00
DEBUG ( 0 , ( " regdb_init: Failed to initialize data in registry! \n " ) ) ;
2008-01-04 17:20:46 +03:00
return false ;
0001-01-01 02:30:17 +02:30
}
2008-01-04 17:20:46 +03:00
return true ;
0001-01-01 02:30:17 +02:30
}
2005-10-07 16:14:25 +04:00
/***********************************************************************
Open the registry . Must already have been initialized by regdb_init ( )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
WERROR regdb_open ( void )
{
WERROR result = WERR_OK ;
if ( tdb_reg ) {
DEBUG ( 10 , ( " regdb_open: incrementing refcount (%d) \n " , tdb_refcount ) ) ;
tdb_refcount + + ;
return WERR_OK ;
}
become_root ( ) ;
2007-11-01 22:53:44 +03:00
tdb_reg = tdb_wrap_open ( NULL , state_path ( " registry.tdb " ) , 0 , REG_TDB_FLAGS , O_RDWR , 0600 ) ;
2005-10-07 16:14:25 +04:00
if ( ! tdb_reg ) {
result = ntstatus_to_werror ( map_nt_error_from_unix ( errno ) ) ;
DEBUG ( 0 , ( " regdb_open: Failed to open %s! (%s) \n " ,
2007-11-01 22:53:44 +03:00
state_path ( " registry.tdb " ) , strerror ( errno ) ) ) ;
2005-10-07 16:14:25 +04:00
}
unbecome_root ( ) ;
tdb_refcount = 1 ;
DEBUG ( 10 , ( " regdb_open: refcount reset (%d) \n " , tdb_refcount ) ) ;
return result ;
}
/***********************************************************************
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
int regdb_close ( void )
{
2008-01-04 14:57:49 +03:00
if ( tdb_refcount = = 0 ) {
return 0 ;
}
2005-10-07 16:14:25 +04:00
tdb_refcount - - ;
DEBUG ( 10 , ( " regdb_close: decrementing refcount (%d) \n " , tdb_refcount ) ) ;
if ( tdb_refcount > 0 )
return 0 ;
SMB_ASSERT ( tdb_refcount > = 0 ) ;
2007-06-15 14:40:36 +04:00
TALLOC_FREE ( tdb_reg ) ;
return 0 ;
2005-10-07 16:14:25 +04:00
}
2007-06-13 17:15:16 +04:00
/***********************************************************************
return the tdb sequence number of the registry tdb .
this is an indicator for the content of the registry
having changed . it will change upon regdb_init , too , though .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
int regdb_get_seqnum ( void )
{
2007-06-15 14:40:36 +04:00
return tdb_get_seqnum ( tdb_reg - > tdb ) ;
2007-06-13 17:15:16 +04:00
}
0001-01-01 02:30:17 +02:30
/***********************************************************************
Add subkey strings to the registry tdb under a defined key
fmt is the same format as tdb_pack except this function only supports
fstrings
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-11-27 04:24:56 +03:00
static bool regdb_store_keys_internal ( const char * key , REGSUBKEY_CTR * ctr )
0001-01-01 02:30:17 +02:30
{
2007-03-27 14:13:53 +04:00
TDB_DATA dbuf ;
2007-11-27 04:24:56 +03:00
uint8 * buffer = NULL ;
0001-01-01 02:30:17 +02:30
int i = 0 ;
uint32 len , buflen ;
2008-01-04 17:20:46 +03:00
bool ret = true ;
2007-11-27 04:24:56 +03:00
uint32 num_subkeys = regsubkey_ctr_numkeys ( ctr ) ;
char * keyname = NULL ;
TALLOC_CTX * ctx = talloc_tos ( ) ;
2005-05-23 20:25:31 +04:00
2007-11-27 04:24:56 +03:00
if ( ! key ) {
return false ;
}
keyname = talloc_strdup ( ctx , key ) ;
if ( ! keyname ) {
return false ;
}
keyname = normalize_reg_path ( ctx , keyname ) ;
2005-06-17 22:57:37 +04:00
0001-01-01 02:30:17 +02:30
/* allocate some initial memory */
2007-11-27 04:24:56 +03:00
if ( ! ( buffer = ( uint8 * ) SMB_MALLOC ( 1024 ) ) ) {
2008-01-04 17:20:46 +03:00
return false ;
2006-07-15 12:36:44 +04:00
}
2007-11-27 04:24:56 +03:00
buflen = 1024 ;
0001-01-01 02:30:17 +02:30
len = 0 ;
2007-11-27 04:24:56 +03:00
0001-01-01 02:30:17 +02:30
/* store the number of subkeys */
2007-11-27 04:24:56 +03:00
0001-01-01 02:30:17 +02:30
len + = tdb_pack ( buffer + len , buflen - len , " d " , num_subkeys ) ;
2007-11-27 04:24:56 +03:00
0001-01-01 02:30:17 +02:30
/* pack all the strings */
2007-11-27 04:24:56 +03:00
0001-01-01 02:30:17 +02:30
for ( i = 0 ; i < num_subkeys ; i + + ) {
0001-01-01 02:30:17 +02:30
len + = tdb_pack ( buffer + len , buflen - len , " f " , regsubkey_ctr_specific_key ( ctr , i ) ) ;
0001-01-01 02:30:17 +02:30
if ( len > buflen ) {
/* allocate some extra space */
2007-03-29 13:35:51 +04:00
if ( ( buffer = ( uint8 * ) SMB_REALLOC ( buffer , len * 2 ) ) = = NULL ) {
2005-06-30 06:59:29 +04:00
DEBUG ( 0 , ( " regdb_store_keys: Failed to realloc memory of size [%d] \n " , len * 2 ) ) ;
2008-01-04 17:20:46 +03:00
ret = false ;
0001-01-01 02:30:17 +02:30
goto done ;
}
buflen = len * 2 ;
2007-11-27 04:24:56 +03:00
0001-01-01 02:30:17 +02:30
len = tdb_pack ( buffer + len , buflen - len , " f " , regsubkey_ctr_specific_key ( ctr , i ) ) ;
2007-11-27 04:24:56 +03:00
}
0001-01-01 02:30:17 +02:30
}
2007-11-27 04:24:56 +03:00
0001-01-01 02:30:17 +02:30
/* finally write out the data */
2007-11-27 04:24:56 +03:00
0001-01-01 02:30:17 +02:30
dbuf . dptr = buffer ;
dbuf . dsize = len ;
2007-06-15 14:40:36 +04:00
if ( tdb_store_bystring ( tdb_reg - > tdb , keyname , dbuf , TDB_REPLACE ) = = - 1 ) {
2008-01-04 17:20:46 +03:00
ret = false ;
0001-01-01 02:30:17 +02:30
goto done ;
}
2007-11-27 04:24:56 +03:00
done :
0001-01-01 02:30:17 +02:30
SAFE_FREE ( buffer ) ;
return ret ;
}
2005-06-17 19:35:31 +04:00
/***********************************************************************
2007-11-27 04:24:56 +03:00
Store the new subkey record and create any child key records that
2005-06-17 19:35:31 +04:00
do not currently exist
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-11-27 04:24:56 +03:00
bool regdb_store_keys ( const char * key , REGSUBKEY_CTR * ctr )
2005-06-17 19:35:31 +04:00
{
int num_subkeys , i ;
2007-11-27 04:24:56 +03:00
char * path = NULL ;
2006-11-22 19:53:28 +03:00
REGSUBKEY_CTR * subkeys = NULL , * old_subkeys = NULL ;
2007-11-27 04:24:56 +03:00
char * oldkeyname = NULL ;
TALLOC_CTX * ctx = talloc_tos ( ) ;
2007-10-06 00:42:14 +04:00
/*
* fetch a list of the old subkeys so we can determine if anything has
* changed
*/
2007-11-27 04:24:56 +03:00
if ( ! ( old_subkeys = TALLOC_ZERO_P ( ctr , REGSUBKEY_CTR ) ) ) {
2007-10-06 00:42:14 +04:00
DEBUG ( 0 , ( " regdb_store_keys: talloc() failure! \n " ) ) ;
2007-11-27 04:24:56 +03:00
return false ;
2007-10-06 00:42:14 +04:00
}
2007-11-27 04:24:56 +03:00
regdb_fetch_keys ( key , old_subkeys ) ;
2007-10-06 00:42:14 +04:00
if ( ctr - > num_subkeys = = old_subkeys - > num_subkeys ) {
for ( i = 0 ; i < ctr - > num_subkeys ; i + + ) {
if ( strcmp ( ctr - > subkeys [ i ] ,
old_subkeys - > subkeys [ i ] ) ! = 0 ) {
break ;
}
}
if ( i = = ctr - > num_subkeys ) {
/*
* Nothing changed , no point to even start a tdb
* transaction
*/
TALLOC_FREE ( old_subkeys ) ;
2007-11-27 04:24:56 +03:00
return true ;
2007-10-06 00:42:14 +04:00
}
}
2007-11-27 04:24:56 +03:00
if ( tdb_transaction_start ( tdb_reg - > tdb ) = = - 1 ) {
2006-11-22 19:53:28 +03:00
DEBUG ( 0 , ( " regdb_store_keys: tdb_transaction_start failed \n " ) ) ;
2007-11-27 04:24:56 +03:00
return false ;
2006-11-22 19:53:28 +03:00
}
2007-10-06 00:42:14 +04:00
/*
* Re - fetch the old keys inside the transaction
*/
TALLOC_FREE ( old_subkeys ) ;
2007-11-27 04:24:56 +03:00
if ( ! ( old_subkeys = TALLOC_ZERO_P ( ctr , REGSUBKEY_CTR ) ) ) {
2005-08-29 18:55:40 +04:00
DEBUG ( 0 , ( " regdb_store_keys: talloc() failure! \n " ) ) ;
2006-11-22 19:53:28 +03:00
goto fail ;
2005-08-29 18:55:40 +04:00
}
2007-11-27 04:24:56 +03:00
regdb_fetch_keys ( key , old_subkeys ) ;
2005-06-17 19:35:31 +04:00
/* store the subkey list for the parent */
2007-11-27 04:24:56 +03:00
if ( ! regdb_store_keys_internal ( key , ctr ) ) {
2006-11-22 18:10:46 +03:00
DEBUG ( 0 , ( " regdb_store_keys: Failed to store new subkey list "
2007-11-27 04:24:56 +03:00
" for parent [%s] \n " , key ) ) ;
2006-11-22 19:53:28 +03:00
goto fail ;
2005-06-17 19:35:31 +04:00
}
2007-11-27 04:24:56 +03:00
2005-06-17 22:57:37 +04:00
/* now delete removed keys */
2006-11-22 19:53:28 +03:00
2007-11-27 04:24:56 +03:00
num_subkeys = regsubkey_ctr_numkeys ( old_subkeys ) ;
for ( i = 0 ; i < num_subkeys ; i + + ) {
oldkeyname = regsubkey_ctr_specific_key ( old_subkeys , i ) ;
if ( regsubkey_ctr_key_exists ( ctr , oldkeyname ) ) {
2006-11-22 19:53:28 +03:00
/*
* It ' s still around , don ' t delete
*/
continue ;
}
2007-11-27 04:24:56 +03:00
path = talloc_asprintf ( ctx , " %s/%s " , key , oldkeyname ) ;
if ( ! path ) {
goto fail ;
}
path = normalize_reg_path ( ctx , path ) ;
if ( ! path ) {
goto fail ;
}
if ( tdb_delete_bystring ( tdb_reg - > tdb , path ) = = - 1 ) {
2006-11-22 19:53:28 +03:00
DEBUG ( 1 , ( " Deleting %s failed \n " , path ) ) ;
goto fail ;
2005-06-17 22:57:37 +04:00
}
2007-11-27 04:24:56 +03:00
TALLOC_FREE ( path ) ;
path = talloc_asprintf ( ctx , " %s/%s/%s " ,
REG_VALUE_PREFIX ,
key ,
oldkeyname ) ;
if ( ! path ) {
goto fail ;
}
path = normalize_reg_path ( ctx , path ) ;
if ( ! path ) {
goto fail ;
}
2006-11-22 19:53:28 +03:00
/*
* Ignore errors here , we might have no values around
*/
2007-06-15 14:40:36 +04:00
tdb_delete_bystring ( tdb_reg - > tdb , path ) ;
2007-11-27 04:24:56 +03:00
TALLOC_FREE ( path ) ;
2005-06-17 22:57:37 +04:00
}
2005-06-27 07:40:03 +04:00
2007-11-27 04:24:56 +03:00
TALLOC_FREE ( old_subkeys ) ;
2005-06-17 19:35:31 +04:00
/* now create records for any subkeys that don't already exist */
2005-08-29 18:55:40 +04:00
2007-11-27 04:24:56 +03:00
num_subkeys = regsubkey_ctr_numkeys ( ctr ) ;
for ( i = 0 ; i < num_subkeys ; i + + ) {
path = talloc_asprintf ( ctx , " %s/%s " ,
key ,
regsubkey_ctr_specific_key ( ctr , i ) ) ;
if ( ! path ) {
goto fail ;
}
if ( ! ( subkeys = TALLOC_ZERO_P ( ctr , REGSUBKEY_CTR ) ) ) {
2005-08-29 18:55:40 +04:00
DEBUG ( 0 , ( " regdb_store_keys: talloc() failure! \n " ) ) ;
2006-11-22 19:53:28 +03:00
goto fail ;
2005-08-29 18:55:40 +04:00
}
2007-11-27 04:24:56 +03:00
if ( regdb_fetch_keys ( path , subkeys ) = = - 1 ) {
2005-06-17 19:35:31 +04:00
/* create a record with 0 subkeys */
2007-11-27 04:24:56 +03:00
if ( ! regdb_store_keys_internal ( path , subkeys ) ) {
2006-11-22 18:10:46 +03:00
DEBUG ( 0 , ( " regdb_store_keys: Failed to store "
2007-11-27 04:24:56 +03:00
" new record for key [%s] \n " , path ) ) ;
2006-11-22 19:53:28 +03:00
goto fail ;
2005-06-17 19:35:31 +04:00
}
}
2005-08-29 18:55:40 +04:00
2007-11-27 04:24:56 +03:00
TALLOC_FREE ( subkeys ) ;
TALLOC_FREE ( path ) ;
2005-06-17 19:35:31 +04:00
}
2006-11-22 19:53:28 +03:00
2007-06-15 14:40:36 +04:00
if ( tdb_transaction_commit ( tdb_reg - > tdb ) = = - 1 ) {
2006-11-22 19:53:28 +03:00
DEBUG ( 0 , ( " regdb_store_keys: Could not commit transaction \n " ) ) ;
2007-11-27 04:24:56 +03:00
return false ;
2006-11-22 19:53:28 +03:00
}
2007-11-27 04:24:56 +03:00
return true ;
2006-11-22 19:53:28 +03:00
2007-06-13 14:12:33 +04:00
fail :
2007-11-27 04:24:56 +03:00
TALLOC_FREE ( old_subkeys ) ;
TALLOC_FREE ( subkeys ) ;
2006-11-22 19:53:28 +03:00
2007-11-27 04:24:56 +03:00
if ( tdb_transaction_cancel ( tdb_reg - > tdb ) = = - 1 ) {
2006-11-22 19:53:28 +03:00
smb_panic ( " regdb_store_keys: tdb_transaction_cancel failed \n " ) ;
}
2007-11-27 04:24:56 +03:00
return false ;
2005-06-17 19:35:31 +04:00
}
0001-01-01 02:30:17 +02:30
/***********************************************************************
2007-11-27 04:24:56 +03:00
Retrieve an array of strings containing subkeys . Memory should be
released by the caller .
0001-01-01 02:30:17 +02:30
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-11-27 04:24:56 +03:00
int regdb_fetch_keys ( const char * key , REGSUBKEY_CTR * ctr )
0001-01-01 02:30:17 +02:30
{
2007-11-27 04:24:56 +03:00
char * path = NULL ;
0001-01-01 02:30:17 +02:30
uint32 num_items ;
TDB_DATA dbuf ;
2007-03-29 13:35:51 +04:00
uint8 * buf ;
0001-01-01 02:30:17 +02:30
uint32 buflen , len ;
int i ;
0001-01-01 02:30:17 +02:30
fstring subkeyname ;
2007-12-14 00:20:58 +03:00
int ret = - 1 ;
TALLOC_CTX * frame = talloc_stackframe ( ) ;
0001-01-01 02:30:17 +02:30
2005-07-15 18:26:11 +04:00
DEBUG ( 11 , ( " regdb_fetch_keys: Enter key => [%s] \n " , key ? key : " NULL " ) ) ;
2007-11-27 04:24:56 +03:00
2007-12-14 00:20:58 +03:00
path = talloc_strdup ( talloc_tos ( ) , key ) ;
2007-11-27 04:24:56 +03:00
if ( ! path ) {
2007-12-14 00:20:58 +03:00
goto fail ;
2007-11-27 04:24:56 +03:00
}
0001-01-01 02:30:17 +02:30
/* convert to key format */
2007-12-14 00:20:58 +03:00
path = talloc_string_sub ( talloc_tos ( ) , path , " \\ " , " / " ) ;
2007-11-27 04:24:56 +03:00
if ( ! path ) {
2007-12-14 00:20:58 +03:00
goto fail ;
2007-11-27 04:24:56 +03:00
}
strupper_m ( path ) ;
2008-01-14 20:31:11 +03:00
if ( tdb_read_lock_bystring_with_timeout ( tdb_reg - > tdb , path , 10 ) = = - 1 ) {
return 0 ;
}
2007-11-27 04:24:56 +03:00
dbuf = tdb_fetch_bystring ( tdb_reg - > tdb , path ) ;
2008-01-14 20:31:11 +03:00
ctr - > seqnum = regdb_get_seqnum ( ) ;
tdb_read_unlock_bystring ( tdb_reg - > tdb , path ) ;
2007-11-27 04:24:56 +03:00
0001-01-01 02:30:17 +02:30
buf = dbuf . dptr ;
buflen = dbuf . dsize ;
2007-11-27 04:24:56 +03:00
0001-01-01 02:30:17 +02:30
if ( ! buf ) {
2005-06-30 06:59:29 +04:00
DEBUG ( 5 , ( " regdb_fetch_keys: tdb lookup failed to locate key [%s] \n " , key ) ) ;
2007-12-14 00:20:58 +03:00
goto fail ;
0001-01-01 02:30:17 +02:30
}
2007-11-27 04:24:56 +03:00
0001-01-01 02:30:17 +02:30
len = tdb_unpack ( buf , buflen , " d " , & num_items ) ;
2007-11-27 04:24:56 +03:00
0001-01-01 02:30:17 +02:30
for ( i = 0 ; i < num_items ; i + + ) {
2007-11-27 04:24:56 +03:00
len + = tdb_unpack ( buf + len , buflen - len , " f " , subkeyname ) ;
regsubkey_ctr_addkey ( ctr , subkeyname ) ;
0001-01-01 02:30:17 +02:30
}
2007-11-27 04:24:56 +03:00
SAFE_FREE ( dbuf . dptr ) ;
2005-07-15 18:26:11 +04:00
DEBUG ( 11 , ( " regdb_fetch_keys: Exit [%d] items \n " , num_items ) ) ;
2007-11-27 04:24:56 +03:00
2007-12-14 00:20:58 +03:00
ret = num_items ;
fail :
TALLOC_FREE ( frame ) ;
return ret ;
0001-01-01 02:30:17 +02:30
}
2005-06-25 21:31:40 +04:00
/****************************************************************************
Unpack a list of registry values frem the TDB
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-11-27 04:24:56 +03:00
2007-03-29 13:35:51 +04:00
static int regdb_unpack_values ( REGVAL_CTR * values , uint8 * buf , int buflen )
2005-06-25 21:31:40 +04:00
{
int len = 0 ;
uint32 type ;
2007-11-27 04:24:56 +03:00
fstring valuename ;
2005-06-29 20:35:32 +04:00
uint32 size ;
2005-06-25 21:31:40 +04:00
uint8 * data_p ;
uint32 num_values = 0 ;
int i ;
2007-11-27 04:24:56 +03:00
2005-06-25 21:31:40 +04:00
/* loop and unpack the rest of the registry values */
2007-11-27 04:24:56 +03:00
2005-06-25 21:31:40 +04:00
len + = tdb_unpack ( buf + len , buflen - len , " d " , & num_values ) ;
2007-11-27 04:24:56 +03:00
2005-06-25 21:31:40 +04:00
for ( i = 0 ; i < num_values ; i + + ) {
/* unpack the next regval */
2007-11-27 04:24:56 +03:00
2005-06-29 20:35:32 +04:00
type = REG_NONE ;
size = 0 ;
data_p = NULL ;
2007-11-27 04:24:56 +03:00
valuename [ 0 ] = ' \0 ' ;
2005-06-25 21:31:40 +04:00
len + = tdb_unpack ( buf + len , buflen - len , " fdB " ,
valuename ,
& type ,
& size ,
& data_p ) ;
2007-11-27 04:24:56 +03:00
2005-06-29 20:35:32 +04:00
/* add the new value. Paranoid protective code -- make sure data_p is valid */
2005-06-25 21:31:40 +04:00
2007-11-27 04:24:56 +03:00
if ( * valuename & & size & & data_p ) {
regval_ctr_addvalue ( values , valuename , type ,
( const char * ) data_p , size ) ;
2005-06-29 20:35:32 +04:00
}
2007-11-27 04:24:56 +03:00
SAFE_FREE ( data_p ) ; /* 'B' option to tdb_unpack does a malloc() */
2005-06-25 21:31:40 +04:00
DEBUG ( 8 , ( " specific: [%s], len: %d \n " , valuename , size ) ) ;
}
return len ;
}
/****************************************************************************
Pack all values in all printer keys
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-11-27 04:24:56 +03:00
2007-03-29 13:35:51 +04:00
static int regdb_pack_values ( REGVAL_CTR * values , uint8 * buf , int buflen )
2005-06-25 21:31:40 +04:00
{
int len = 0 ;
int i ;
REGISTRY_VALUE * val ;
2006-03-12 20:53:57 +03:00
int num_values ;
2005-06-25 21:31:40 +04:00
if ( ! values )
return 0 ;
2006-03-12 20:53:57 +03:00
num_values = regval_ctr_numvals ( values ) ;
2005-06-25 21:31:40 +04:00
/* pack the number of values first */
2007-11-27 04:24:56 +03:00
2005-06-25 21:31:40 +04:00
len + = tdb_pack ( buf + len , buflen - len , " d " , num_values ) ;
2007-11-27 04:24:56 +03:00
2005-06-25 21:31:40 +04:00
/* loop over all values */
2007-11-27 04:24:56 +03:00
for ( i = 0 ; i < num_values ; i + + ) {
2005-06-25 21:31:40 +04:00
val = regval_ctr_specific_value ( values , i ) ;
len + = tdb_pack ( buf + len , buflen - len , " fdB " ,
regval_name ( val ) ,
regval_type ( val ) ,
regval_size ( val ) ,
regval_data_p ( val ) ) ;
}
return len ;
}
0001-01-01 02:30:17 +02:30
0001-01-01 02:30:17 +02:30
/***********************************************************************
2007-11-27 04:24:56 +03:00
Retrieve an array of strings containing subkeys . Memory should be
2005-06-25 21:31:40 +04:00
released by the caller .
0001-01-01 02:30:17 +02:30
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-06-30 06:59:29 +04:00
int regdb_fetch_values ( const char * key , REGVAL_CTR * values )
0001-01-01 02:30:17 +02:30
{
2005-06-25 21:31:40 +04:00
TDB_DATA data ;
2007-11-27 04:24:56 +03:00
char * keystr = NULL ;
TALLOC_CTX * ctx = talloc_tos ( ) ;
2005-05-09 17:51:44 +04:00
2005-06-30 06:59:29 +04:00
DEBUG ( 10 , ( " regdb_fetch_values: Looking for value of key [%s] \n " , key ) ) ;
2007-11-27 04:24:56 +03:00
keystr = talloc_asprintf ( ctx , " %s/%s " , REG_VALUE_PREFIX , key ) ;
if ( ! keystr ) {
return 0 ;
}
keystr = normalize_reg_path ( ctx , keystr ) ;
if ( ! keystr ) {
return 0 ;
}
2008-01-14 20:31:11 +03:00
if ( tdb_read_lock_bystring_with_timeout ( tdb_reg - > tdb , keystr , 10 ) = = - 1 ) {
return 0 ;
}
2007-11-27 04:24:56 +03:00
data = tdb_fetch_bystring ( tdb_reg - > tdb , keystr ) ;
2008-01-14 20:31:11 +03:00
values - > seqnum = regdb_get_seqnum ( ) ;
tdb_read_unlock_bystring ( tdb_reg - > tdb , keystr ) ;
2007-11-27 04:24:56 +03:00
if ( ! data . dptr ) {
2005-06-27 07:40:03 +04:00
/* all keys have zero values by default */
2005-06-25 21:31:40 +04:00
return 0 ;
2005-06-27 07:40:03 +04:00
}
2007-11-27 04:24:56 +03:00
regdb_unpack_values ( values , data . dptr , data . dsize ) ;
SAFE_FREE ( data . dptr ) ;
2005-06-25 21:31:40 +04:00
return regval_ctr_numvals ( values ) ;
0001-01-01 02:30:17 +02:30
}
2007-10-19 04:40:25 +04:00
bool regdb_store_values ( const char * key , REGVAL_CTR * values )
0001-01-01 02:30:17 +02:30
{
2007-10-06 00:42:14 +04:00
TDB_DATA old_data , data ;
2007-11-27 04:24:56 +03:00
char * keystr = NULL ;
TALLOC_CTX * ctx = talloc_tos ( ) ;
2005-06-25 21:31:40 +04:00
int len , ret ;
2007-11-27 04:24:56 +03:00
2005-06-30 06:59:29 +04:00
DEBUG ( 10 , ( " regdb_store_values: Looking for value of key [%s] \n " , key ) ) ;
2007-11-27 04:24:56 +03:00
ZERO_STRUCT ( data ) ;
len = regdb_pack_values ( values , data . dptr , data . dsize ) ;
if ( len < = 0 ) {
2005-06-30 06:59:29 +04:00
DEBUG ( 0 , ( " regdb_store_values: unable to pack values. len <= 0 \n " ) ) ;
2007-11-27 04:24:56 +03:00
return false ;
2005-06-25 21:31:40 +04:00
}
2007-11-27 04:24:56 +03:00
2007-03-29 13:35:51 +04:00
data . dptr = SMB_MALLOC_ARRAY ( uint8 , len ) ;
2005-06-25 21:31:40 +04:00
data . dsize = len ;
2007-11-27 04:24:56 +03:00
len = regdb_pack_values ( values , data . dptr , data . dsize ) ;
2005-06-25 21:31:40 +04:00
SMB_ASSERT ( len = = data . dsize ) ;
2007-11-27 04:24:56 +03:00
keystr = talloc_asprintf ( ctx , " %s/%s " , REG_VALUE_PREFIX , key ) ;
if ( ! keystr ) {
SAFE_FREE ( data . dptr ) ;
return false ;
}
keystr = normalize_reg_path ( ctx , keystr ) ;
if ( ! keystr ) {
SAFE_FREE ( data . dptr ) ;
return false ;
}
2007-10-06 00:42:14 +04:00
old_data = tdb_fetch_bystring ( tdb_reg - > tdb , keystr ) ;
if ( ( old_data . dptr ! = NULL )
& & ( old_data . dsize = = data . dsize )
& & ( memcmp ( old_data . dptr , data . dptr , data . dsize ) = = 0 ) ) {
SAFE_FREE ( old_data . dptr ) ;
SAFE_FREE ( data . dptr ) ;
2008-01-04 17:20:46 +03:00
return true ;
2007-10-06 00:42:14 +04:00
}
2007-06-15 14:40:36 +04:00
ret = tdb_trans_store_bystring ( tdb_reg - > tdb , keystr , data , TDB_REPLACE ) ;
2007-11-27 04:24:56 +03:00
2007-10-06 00:42:14 +04:00
SAFE_FREE ( old_data . dptr ) ;
2005-06-25 21:31:40 +04:00
SAFE_FREE ( data . dptr ) ;
2007-11-27 04:24:56 +03:00
2005-06-25 21:31:40 +04:00
return ret ! = - 1 ;
0001-01-01 02:30:17 +02:30
}
2006-11-30 10:38:40 +03:00
static WERROR regdb_get_secdesc ( TALLOC_CTX * mem_ctx , const char * key ,
struct security_descriptor * * psecdesc )
{
char * tdbkey ;
TDB_DATA data ;
NTSTATUS status ;
DEBUG ( 10 , ( " regdb_get_secdesc: Getting secdesc of key [%s] \n " , key ) ) ;
2007-06-16 03:47:40 +04:00
if ( asprintf ( & tdbkey , " %s/%s " , REG_SECDESC_PREFIX , key ) = = - 1 ) {
2006-11-30 10:38:40 +03:00
return WERR_NOMEM ;
}
normalize_dbkey ( tdbkey ) ;
2007-06-15 14:40:36 +04:00
data = tdb_fetch_bystring ( tdb_reg - > tdb , tdbkey ) ;
2006-11-30 10:38:40 +03:00
SAFE_FREE ( tdbkey ) ;
if ( data . dptr = = NULL ) {
return WERR_BADFILE ;
}
status = unmarshall_sec_desc ( mem_ctx , ( uint8 * ) data . dptr , data . dsize ,
psecdesc ) ;
SAFE_FREE ( data . dptr ) ;
if ( NT_STATUS_EQUAL ( status , NT_STATUS_NO_MEMORY ) ) {
return WERR_NOMEM ;
}
if ( ! NT_STATUS_IS_OK ( status ) ) {
return WERR_REG_CORRUPT ;
}
return WERR_OK ;
}
static WERROR regdb_set_secdesc ( const char * key ,
struct security_descriptor * secdesc )
{
prs_struct ps ;
TALLOC_CTX * mem_ctx ;
char * tdbkey ;
WERROR err = WERR_NOMEM ;
TDB_DATA tdbdata ;
if ( ! ( mem_ctx = talloc_init ( " regdb_set_secdesc " ) ) ) {
return WERR_NOMEM ;
}
ZERO_STRUCT ( ps ) ;
2007-06-16 03:47:40 +04:00
if ( ! ( tdbkey = talloc_asprintf ( mem_ctx , " %s/%s " , REG_SECDESC_PREFIX ,
2006-11-30 10:38:40 +03:00
key ) ) ) {
goto done ;
}
normalize_dbkey ( tdbkey ) ;
2007-11-06 02:50:47 +03:00
if ( secdesc = = NULL ) {
/* assuming a delete */
int tdb_ret ;
tdb_ret = tdb_trans_delete ( tdb_reg - > tdb ,
string_term_tdb_data ( tdbkey ) ) ;
if ( tdb_ret = = - 1 ) {
err = ntstatus_to_werror ( map_nt_error_from_unix ( errno ) ) ;
} else {
err = WERR_OK ;
}
goto done ;
}
2007-03-27 14:05:20 +04:00
err = ntstatus_to_werror ( marshall_sec_desc ( mem_ctx , secdesc ,
2007-06-13 14:07:05 +04:00
& tdbdata . dptr ,
2006-11-30 10:38:40 +03:00
& tdbdata . dsize ) ) ;
if ( ! W_ERROR_IS_OK ( err ) ) {
goto done ;
}
2007-06-15 14:40:36 +04:00
if ( tdb_trans_store_bystring ( tdb_reg - > tdb , tdbkey , tdbdata , 0 ) = = - 1 ) {
2006-11-30 10:38:40 +03:00
err = ntstatus_to_werror ( map_nt_error_from_unix ( errno ) ) ;
goto done ;
}
done :
prs_mem_free ( & ps ) ;
TALLOC_FREE ( mem_ctx ) ;
return err ;
}
0001-01-01 02:30:17 +02:30
2008-01-14 20:31:11 +03:00
bool regdb_subkeys_need_update ( REGSUBKEY_CTR * subkeys )
{
return ( regdb_get_seqnum ( ) ! = subkeys - > seqnum ) ;
}
bool regdb_values_need_update ( REGVAL_CTR * values )
{
return ( regdb_get_seqnum ( ) ! = values - > seqnum ) ;
}
0001-01-01 02:30:17 +02:30
/*
* Table of function pointers for default access
*/
REGISTRY_OPS regdb_ops = {
2005-06-30 06:59:29 +04:00
regdb_fetch_keys ,
regdb_fetch_values ,
regdb_store_keys ,
regdb_store_values ,
2006-11-30 10:38:40 +03:00
NULL ,
regdb_get_secdesc ,
2008-01-14 20:31:11 +03:00
regdb_set_secdesc ,
regdb_subkeys_need_update ,
regdb_values_need_update
0001-01-01 02:30:17 +02:30
} ;