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 ,
2005-08-05 18:34:25 +04:00
" HKLM \\ SOFTWARE \\ Microsoft \\ Windows NT \\ CurrentVersion \\ Perflib " ,
" HKLM \\ SOFTWARE \\ Microsoft \\ Windows NT \\ CurrentVersion \\ Perflib \\ 009 " ,
2005-07-02 02:24:00 +04:00
" HKLM \\ SYSTEM \\ CurrentControlSet \\ Control \\ Print \\ Monitors " ,
2005-05-23 20:25:31 +04:00
" HKLM \\ SYSTEM \\ CurrentControlSet \\ Control \\ ProductOptions " ,
2005-09-01 18:58:57 +04:00
" HKLM \\ SYSTEM \\ CurrentControlSet \\ Control \\ Terminal Server \\ DefaultUserConfiguration " ,
2005-05-23 20:25:31 +04:00
" HKLM \\ SYSTEM \\ CurrentControlSet \\ Services \\ TcpIp \\ Parameters " ,
" HKLM \\ SYSTEM \\ CurrentControlSet \\ Services \\ Netlogon \\ Parameters " ,
2006-12-03 20:34:11 +03:00
" HKU " ,
" HKCR " ,
" HKPD " ,
" 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-10-19 04:40:25 +04:00
static bool init_registry_data ( void )
0001-01-01 02:30:17 +02:30
{
2006-12-03 20:34:11 +03:00
pstring path , base , remaining ;
fstring keyname , subkeyname ;
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 " ) ) ;
return False ;
}
2005-05-23 20:25:31 +04:00
/* loop over all of the predefined paths and add each component */
for ( i = 0 ; builtin_registry_paths [ i ] ! = NULL ; i + + ) {
2006-12-03 20:34:11 +03:00
DEBUG ( 6 , ( " init_registry_data: Adding [%s] \n " , builtin_registry_paths [ i ] ) ) ;
pstrcpy ( path , builtin_registry_paths [ i ] ) ;
pstrcpy ( base , " " ) ;
p = path ;
while ( next_token ( & p , keyname , " \\ " , sizeof ( keyname ) ) ) {
/* build up the registry path from the components */
if ( * base )
pstrcat ( base , " \\ " ) ;
pstrcat ( base , keyname ) ;
/* get the immediate subkeyname (if we have one ) */
* subkeyname = ' \0 ' ;
if ( * p ) {
pstrcpy ( remaining , p ) ;
p2 = remaining ;
if ( ! next_token ( & p2 , subkeyname , " \\ " , sizeof ( subkeyname ) ) )
fstrcpy ( subkeyname , p2 ) ;
}
DEBUG ( 10 , ( " init_registry_data: Storing key [%s] with subkey [%s] \n " ,
base , * subkeyname ? subkeyname : " NULL " ) ) ;
/* we don't really care if the lookup succeeds or not since
we are about to update the record . We just want any
subkeys already present */
if ( ! ( subkeys = TALLOC_ZERO_P ( NULL , REGSUBKEY_CTR ) ) ) {
DEBUG ( 0 , ( " talloc() failure! \n " ) ) ;
2007-02-07 16:26:13 +03:00
goto fail ;
2006-12-03 20:34:11 +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 ;
2006-12-03 20:34:11 +03:00
TALLOC_FREE ( subkeys ) ;
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 */
for ( i = 0 ; builtin_registry_values [ i ] . path ! = NULL ; i + + ) {
2006-12-03 20:34:11 +03:00
if ( ! ( values = TALLOC_ZERO_P ( NULL , REGVAL_CTR ) ) ) {
DEBUG ( 0 , ( " talloc() failure! \n " ) ) ;
2007-02-07 16:26:13 +03:00
goto fail ;
2005-08-29 18:55:40 +04:00
}
2006-12-03 20:34:11 +03:00
regdb_fetch_values ( builtin_registry_values [ i ] . path , values ) ;
/* preserve existing values across restarts. Only add new ones */
if ( ! regval_ctr_key_exists ( values , builtin_registry_values [ i ] . valuename ) )
{
switch ( builtin_registry_values [ i ] . type ) {
case REG_DWORD :
regval_ctr_addvalue ( values ,
builtin_registry_values [ i ] . valuename ,
REG_DWORD ,
( char * ) & builtin_registry_values [ i ] . data . dw_value ,
sizeof ( uint32 ) ) ;
break ;
case REG_SZ :
init_unistr2 ( & data , builtin_registry_values [ i ] . data . string , UNI_STR_TERMINATE ) ;
regval_ctr_addvalue ( values ,
builtin_registry_values [ i ] . valuename ,
REG_SZ ,
( char * ) data . buffer ,
data . uni_str_len * sizeof ( uint16 ) ) ;
break ;
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-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 " ) ) ;
return False ;
}
0001-01-01 02:30:17 +02:30
return True ;
2007-02-07 16:26:13 +03:00
fail :
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 " ) ;
}
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
2005-06-17 19:53:01 +04:00
if ( tdb_reg )
0001-01-01 02:30:17 +02:30
return True ;
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 ) ) ) ;
0001-01-01 02:30:17 +02:30
return False ;
}
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 ( ) ) {
2006-07-15 12:36:44 +04:00
DEBUG ( 0 , ( " init_registry: Failed to initialize data in registry! \n " ) ) ;
2005-09-01 18:00:53 +04:00
return False ;
0001-01-01 02:30:17 +02:30
}
return True ;
}
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 )
{
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-10-19 04:40:25 +04: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-03-29 13:35:51 +04:00
uint8 * buffer ;
0001-01-01 02:30:17 +02:30
int i = 0 ;
uint32 len , buflen ;
2007-10-19 04:40:25 +04:00
bool ret = True ;
0001-01-01 02:30:17 +02:30
uint32 num_subkeys = regsubkey_ctr_numkeys ( ctr ) ;
2005-05-23 20:25:31 +04:00
pstring keyname ;
0001-01-01 02:30:17 +02:30
2005-05-23 20:25:31 +04:00
if ( ! key )
0001-01-01 02:30:17 +02:30
return False ;
2005-05-23 20:25:31 +04:00
pstrcpy ( keyname , key ) ;
2005-06-17 22:57:37 +04:00
normalize_reg_path ( keyname ) ;
0001-01-01 02:30:17 +02:30
/* allocate some initial memory */
2007-03-29 13:35:51 +04:00
if ( ! ( buffer = ( uint8 * ) SMB_MALLOC ( sizeof ( pstring ) ) ) ) {
2006-07-15 12:36:44 +04:00
return False ;
}
0001-01-01 02:30:17 +02:30
buflen = sizeof ( pstring ) ;
len = 0 ;
/* store the number of subkeys */
0001-01-01 02:30:17 +02:30
len + = tdb_pack ( buffer + len , buflen - len , " d " , num_subkeys ) ;
0001-01-01 02:30:17 +02:30
/* pack all the strings */
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 ) ) ;
0001-01-01 02:30:17 +02:30
ret = False ;
goto done ;
}
buflen = len * 2 ;
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
}
}
/* finally write out the data */
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 ) {
0001-01-01 02:30:17 +02:30
ret = False ;
goto done ;
}
done :
SAFE_FREE ( buffer ) ;
0001-01-01 02:30:17 +02:30
0001-01-01 02:30:17 +02:30
return ret ;
}
2005-06-17 19:35:31 +04:00
/***********************************************************************
Store the new subkey record and create any child key records that
do not currently exist
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-19 04:40:25 +04:00
bool regdb_store_keys ( const char * key , REGSUBKEY_CTR * ctr )
2005-06-17 19:35:31 +04:00
{
int num_subkeys , i ;
pstring path ;
2006-11-22 19:53:28 +03:00
REGSUBKEY_CTR * subkeys = NULL , * old_subkeys = NULL ;
2005-06-17 22:57:37 +04:00
char * oldkeyname ;
2007-10-06 00:42:14 +04:00
/*
* fetch a list of the old subkeys so we can determine if anything has
* changed
*/
if ( ! ( old_subkeys = TALLOC_ZERO_P ( ctr , REGSUBKEY_CTR ) ) ) {
DEBUG ( 0 , ( " regdb_store_keys: talloc() failure! \n " ) ) ;
goto fail ;
}
regdb_fetch_keys ( key , old_subkeys ) ;
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 ) ;
return True ;
}
}
2007-06-15 14:40:36 +04: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 " ) ) ;
return False ;
}
2007-10-06 00:42:14 +04:00
/*
* Re - fetch the old keys inside the transaction
*/
TALLOC_FREE ( old_subkeys ) ;
2005-08-29 18:55:40 +04:00
if ( ! ( old_subkeys = TALLOC_ZERO_P ( ctr , REGSUBKEY_CTR ) ) ) {
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
}
regdb_fetch_keys ( key , old_subkeys ) ;
2005-06-17 19:35:31 +04:00
/* store the subkey list for the parent */
2005-06-30 06:59:29 +04: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 "
2006-11-22 19:53:28 +03:00
" for parent [%s] \n " , key ) ) ;
goto fail ;
2005-06-17 19:35:31 +04:00
}
2005-06-17 22:57:37 +04:00
/* now delete removed keys */
2005-08-29 18:55:40 +04:00
num_subkeys = regsubkey_ctr_numkeys ( old_subkeys ) ;
2005-06-17 22:57:37 +04:00
for ( i = 0 ; i < num_subkeys ; i + + ) {
2005-08-29 18:55:40 +04:00
oldkeyname = regsubkey_ctr_specific_key ( old_subkeys , i ) ;
2006-11-22 19:53:28 +03:00
if ( regsubkey_ctr_key_exists ( ctr , oldkeyname ) ) {
/*
* It ' s still around , don ' t delete
*/
continue ;
}
pstr_sprintf ( path , " %s/%s " , key , oldkeyname ) ;
normalize_reg_path ( path ) ;
2007-06-15 14:40:36 +04:00
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
}
2006-11-22 19:53:28 +03:00
2007-06-16 03:47:40 +04:00
pstr_sprintf ( path , " %s/%s/%s " , REG_VALUE_PREFIX , key ,
2006-11-22 19:53:28 +03:00
oldkeyname ) ;
normalize_reg_path ( path ) ;
/*
* Ignore errors here , we might have no values around
*/
2007-06-15 14:40:36 +04:00
tdb_delete_bystring ( tdb_reg - > tdb , path ) ;
2005-06-17 22:57:37 +04:00
}
2005-06-27 07:40:03 +04:00
2005-08-29 18:55:40 +04:00
TALLOC_FREE ( old_subkeys ) ;
2005-06-17 22:57:37 +04:00
2005-06-17 19:35:31 +04:00
/* now create records for any subkeys that don't already exist */
num_subkeys = regsubkey_ctr_numkeys ( ctr ) ;
for ( i = 0 ; i < num_subkeys ; i + + ) {
2006-11-22 19:53:28 +03:00
pstr_sprintf ( path , " %s/%s " , key ,
2006-11-22 18:10:46 +03:00
regsubkey_ctr_specific_key ( ctr , i ) ) ;
2005-08-29 18:55:40 +04:00
if ( ! ( subkeys = TALLOC_ZERO_P ( ctr , REGSUBKEY_CTR ) ) ) {
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
}
if ( regdb_fetch_keys ( path , subkeys ) = = - 1 ) {
2005-06-17 19:35:31 +04:00
/* create a record with 0 subkeys */
2005-08-29 18:55:40 +04: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-06-13 14:06:17 +04: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
TALLOC_FREE ( subkeys ) ;
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 " ) ) ;
return False ;
}
2005-06-17 19:35:31 +04:00
return True ;
2006-11-22 19:53:28 +03:00
2007-06-13 14:12:33 +04:00
fail :
2006-11-22 19:53:28 +03:00
TALLOC_FREE ( old_subkeys ) ;
TALLOC_FREE ( subkeys ) ;
2007-06-15 14:40:36 +04: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 " ) ;
}
return False ;
2005-06-17 19:35:31 +04:00
}
0001-01-01 02:30:17 +02:30
/***********************************************************************
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_keys ( const char * key , REGSUBKEY_CTR * ctr )
0001-01-01 02:30:17 +02:30
{
pstring path ;
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 ;
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 " ) ) ;
0001-01-01 02:30:17 +02:30
pstrcpy ( path , key ) ;
/* convert to key format */
0001-01-01 02:30:17 +02:30
pstring_sub ( path , " \\ " , " / " ) ;
0001-01-01 02:30:17 +02:30
strupper_m ( path ) ;
0001-01-01 02:30:17 +02:30
2007-06-15 14:40:36 +04:00
dbuf = tdb_fetch_bystring ( tdb_reg - > tdb , path ) ;
0001-01-01 02:30:17 +02:30
buf = dbuf . dptr ;
buflen = dbuf . dsize ;
if ( ! buf ) {
2005-06-30 06:59:29 +04:00
DEBUG ( 5 , ( " regdb_fetch_keys: tdb lookup failed to locate key [%s] \n " , key ) ) ;
0001-01-01 02:30:17 +02:30
return - 1 ;
0001-01-01 02:30:17 +02:30
}
len = tdb_unpack ( buf , buflen , " d " , & num_items ) ;
for ( i = 0 ; i < num_items ; i + + ) {
0001-01-01 02:30:17 +02:30
len + = tdb_unpack ( buf + len , buflen - len , " f " , subkeyname ) ;
regsubkey_ctr_addkey ( ctr , subkeyname ) ;
0001-01-01 02:30:17 +02:30
}
0001-01-01 02:30:17 +02:30
SAFE_FREE ( dbuf . dptr ) ;
2005-07-15 18:26:11 +04:00
DEBUG ( 11 , ( " regdb_fetch_keys: Exit [%d] items \n " , num_items ) ) ;
0001-01-01 02:30:17 +02:30
0001-01-01 02:30:17 +02:30
return num_items ;
}
2005-06-25 21:31:40 +04:00
/****************************************************************************
Unpack a list of registry values frem the TDB
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
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 ;
pstring 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 ;
/* loop and unpack the rest of the registry values */
len + = tdb_unpack ( buf + len , buflen - len , " d " , & num_values ) ;
for ( i = 0 ; i < num_values ; i + + ) {
/* unpack the next regval */
2005-06-29 20:35:32 +04:00
type = REG_NONE ;
size = 0 ;
data_p = NULL ;
2005-06-25 21:31:40 +04:00
len + = tdb_unpack ( buf + len , buflen - len , " fdB " ,
valuename ,
& type ,
& size ,
& data_p ) ;
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
2005-06-29 20:35:32 +04:00
if ( size & & data_p ) {
regval_ctr_addvalue ( values , valuename , type , ( const char * ) data_p , size ) ;
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-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 */
len + = tdb_pack ( buf + len , buflen - len , " d " , num_values ) ;
/* loop over all values */
for ( i = 0 ; i < num_values ; i + + ) {
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
/***********************************************************************
0001-01-01 02:30:17 +02:30
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 ;
pstring keystr ;
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 ) ) ;
2005-06-25 21:31:40 +04:00
2007-06-16 03:47:40 +04:00
pstr_sprintf ( keystr , " %s/%s " , REG_VALUE_PREFIX , key ) ;
2005-06-25 21:31:40 +04:00
normalize_reg_path ( keystr ) ;
2007-06-15 14:40:36 +04:00
data = tdb_fetch_bystring ( tdb_reg - > tdb , keystr ) ;
2005-06-25 21:31:40 +04:00
2005-06-27 07:40:03 +04:00
if ( ! data . dptr ) {
/* 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
}
2005-06-25 21:31:40 +04:00
2005-08-13 01:45:58 +04:00
regdb_unpack_values ( values , data . dptr , data . dsize ) ;
2005-06-25 21:31:40 +04:00
SAFE_FREE ( data . dptr ) ;
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 ;
2005-06-25 21:31:40 +04:00
pstring keystr ;
int len , ret ;
2005-06-30 06:59:29 +04:00
DEBUG ( 10 , ( " regdb_store_values: Looking for value of key [%s] \n " , key ) ) ;
2005-06-25 21:31:40 +04: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 " ) ) ;
2005-06-25 21:31:40 +04:00
return False ;
}
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 ;
len = regdb_pack_values ( values , data . dptr , data . dsize ) ;
SMB_ASSERT ( len = = data . dsize ) ;
2007-06-16 03:47:40 +04:00
pstr_sprintf ( keystr , " %s/%s " , REG_VALUE_PREFIX , key ) ;
2005-06-25 21:31:40 +04:00
normalize_reg_path ( keystr ) ;
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 ) ;
return True ;
}
2007-06-15 14:40:36 +04:00
ret = tdb_trans_store_bystring ( tdb_reg - > tdb , keystr , data , TDB_REPLACE ) ;
2005-06-25 21:31:40 +04: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 ) ;
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-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
/*
* 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 ,
regdb_set_secdesc
0001-01-01 02:30:17 +02:30
} ;