2002-07-18 23:00:24 +00:00
/*
* Unix SMB / CIFS implementation .
2005-05-23 16:25:31 +00:00
* Virtual Windows Registry Layer
* Copyright ( C ) Gerald Carter 2002 - 2005
2009-07-07 17:16:21 +02:00
* Copyright ( C ) Michael Adam 2007 - 2009
2002-07-18 23:00:24 +00: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
2007-07-09 19:25:36 +00:00
* the Free Software Foundation ; either version 3 of the License , or
2002-07-18 23:00:24 +00:00
* ( 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 05:23:25 +00:00
* along with this program ; if not , see < http : //www.gnu.org/licenses/>.
2002-07-18 23:00:24 +00:00
*/
/* Implementation of internal registry database functions. */
# include "includes.h"
# undef DBGC_CLASS
2007-09-28 23:05:52 +00:00
# define DBGC_CLASS DBGC_REGISTRY
2002-07-18 23:00:24 +00:00
2008-03-16 23:07:15 +01:00
static struct db_context * regdb = NULL ;
2008-03-16 23:08:33 +01:00
static int regdb_refcount ;
2002-07-18 23:00:24 +00:00
2009-07-03 17:39:17 +02:00
static bool regdb_key_exists ( struct db_context * db , const char * key ) ;
2008-05-08 00:34:35 +02:00
static bool regdb_key_is_base_key ( const char * key ) ;
2009-07-15 12:45:43 +02:00
static WERROR regdb_fetch_keys_internal ( struct db_context * db , const char * key ,
struct regsubkey_ctr * ctr ) ;
2009-07-07 11:31:28 +02:00
static bool regdb_store_keys_internal ( struct db_context * db , const char * key ,
struct regsubkey_ctr * ctr ) ;
2009-07-08 12:32:48 +02:00
static int regdb_fetch_values_internal ( struct db_context * db , const char * key ,
struct regval_ctr * values ) ;
2009-07-08 12:38:41 +02:00
static bool regdb_store_values_internal ( struct db_context * db , const char * key ,
struct regval_ctr * values ) ;
2008-04-29 17:17:02 +02:00
2005-05-23 16:25:31 +00:00
/* List the deepest path into the registry. All part components will be created.*/
2005-08-29 14:55:40 +00:00
/* If you want to have a part of the path controlled by the tdb and part by
2005-05-23 16:25:31 +00: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 19:43:53 +00:00
KEY_PRINTING_2K ,
KEY_PRINTING_PORTS ,
KEY_PRINTING ,
KEY_SHARES ,
KEY_EVENTLOG ,
2006-11-30 07:38:40 +00:00
KEY_SMBCONF ,
2008-01-20 03:39:27 +01:00
KEY_PERFLIB ,
KEY_PERFLIB_009 ,
2008-03-18 21:30:34 +01:00
KEY_GROUP_POLICY ,
KEY_SAMBA_GROUP_POLICY ,
KEY_GP_MACHINE_POLICY ,
KEY_GP_MACHINE_WIN_POLICY ,
KEY_HKCU ,
KEY_GP_USER_POLICY ,
KEY_GP_USER_WIN_POLICY ,
KEY_WINLOGON_GPEXT_PATH ,
2005-07-01 22:24:00 +00:00
" HKLM \\ SYSTEM \\ CurrentControlSet \\ Control \\ Print \\ Monitors " ,
2008-01-18 16:15:43 +01:00
KEY_PROD_OPTIONS ,
2005-09-01 14:58:57 +00:00
" HKLM \\ SYSTEM \\ CurrentControlSet \\ Control \\ Terminal Server \\ DefaultUserConfiguration " ,
2008-01-19 23:46:13 +01:00
KEY_TCPIP_PARAMS ,
2008-01-18 16:13:01 +01:00
KEY_NETLOGON_PARAMS ,
2008-01-20 01:09:52 +01:00
KEY_HKU ,
KEY_HKCR ,
KEY_HKPD ,
KEY_HKPT ,
2005-05-23 16:25:31 +00:00
NULL } ;
2005-06-17 15:53:01 +00:00
2005-06-24 22:34:40 +00:00
struct builtin_regkey_value {
const char * path ;
const char * valuename ;
uint32 type ;
union {
const char * string ;
uint32 dw_value ;
} data ;
} ;
2005-06-25 17:31:40 +00:00
static struct builtin_regkey_value builtin_registry_values [ ] = {
2005-08-05 14:34:25 +00:00
{ KEY_PRINTING_PORTS ,
2005-07-04 16:52:29 +00:00
SAMBA_PRINTER_PORT_NAME , REG_SZ , { " " } } ,
2005-08-05 14:34:25 +00:00
{ KEY_PRINTING_2K ,
2005-07-04 16:52:29 +00:00
" DefaultSpoolDirectory " , REG_SZ , { " C: \\ Windows \\ System32 \\ Spool \\ Printers " } } ,
2005-08-05 14:34:25 +00:00
{ KEY_EVENTLOG ,
" DisplayName " , REG_SZ , { " Event Log " } } ,
{ KEY_EVENTLOG ,
" ErrorControl " , REG_DWORD , { ( char * ) 0x00000001 } } ,
2005-06-25 17:31:40 +00:00
{ NULL , NULL , 0 , { NULL } }
2005-06-24 22:34:40 +00:00
} ;
2008-03-20 13:59:09 +01:00
/**
* Initialize a key in the registry :
* create each component key of the specified path .
*/
2009-07-07 23:03:46 +02:00
static WERROR init_registry_key_internal ( struct db_context * db ,
const char * add_path )
2002-07-18 23:00:24 +00:00
{
2008-04-13 13:38:44 +02:00
WERROR werr ;
2008-03-20 13:59:09 +01:00
TALLOC_CTX * frame = talloc_stackframe ( ) ;
2007-11-26 17:24:56 -08:00
char * path = NULL ;
char * base = NULL ;
char * remaining = NULL ;
2007-12-07 17:32:32 -08:00
char * keyname ;
char * subkeyname ;
2009-02-24 15:19:18 +01:00
struct regsubkey_ctr * subkeys ;
2006-12-03 17:34:11 +00:00
const char * p , * p2 ;
2007-02-07 13:26:13 +00:00
2008-03-20 13:59:09 +01:00
DEBUG ( 6 , ( " init_registry_key: Adding [%s] \n " , add_path ) ) ;
2007-02-07 13:26:13 +00:00
2008-03-20 13:59:09 +01:00
path = talloc_strdup ( frame , add_path ) ;
base = talloc_strdup ( frame , " " ) ;
if ( ! path | | ! base ) {
2008-04-13 13:38:44 +02:00
werr = WERR_NOMEM ;
2008-03-20 13:59:09 +01:00
goto fail ;
2007-02-07 13:26:13 +00:00
}
2008-03-20 13:59:09 +01:00
p = path ;
2007-11-26 17:24:56 -08:00
2008-03-20 13:59:09 +01:00
while ( next_token_talloc ( frame , & p , & keyname , " \\ " ) ) {
2006-12-03 17:34:11 +00:00
2008-03-20 13:59:09 +01:00
/* build up the registry path from the components */
2007-11-26 17:24:56 -08:00
2008-03-20 13:59:09 +01:00
if ( * base ) {
base = talloc_asprintf ( frame , " %s \\ " , base ) ;
2007-11-26 17:24:56 -08:00
if ( ! base ) {
2008-04-13 13:38:44 +02:00
werr = WERR_NOMEM ;
2007-11-26 17:24:56 -08:00
goto fail ;
}
2008-03-20 13:59:09 +01:00
}
base = talloc_asprintf_append ( base , " %s " , keyname ) ;
if ( ! base ) {
2008-04-13 13:38:44 +02:00
werr = WERR_NOMEM ;
2008-03-20 13:59:09 +01:00
goto fail ;
}
2007-11-26 17:24:56 -08:00
2008-03-20 13:59:09 +01:00
/* get the immediate subkeyname (if we have one ) */
2007-11-26 17:24:56 -08:00
2008-03-20 13:59:09 +01:00
subkeyname = talloc_strdup ( frame , " " ) ;
if ( ! subkeyname ) {
2008-04-13 13:38:44 +02:00
werr = WERR_NOMEM ;
2008-03-20 13:59:09 +01:00
goto fail ;
}
if ( * p ) {
remaining = talloc_strdup ( frame , p ) ;
if ( ! remaining ) {
2008-04-13 13:38:44 +02:00
werr = WERR_NOMEM ;
2007-12-07 17:32:32 -08:00
goto fail ;
}
2008-03-20 13:59:09 +01:00
p2 = remaining ;
if ( ! next_token_talloc ( frame , & p2 ,
& subkeyname , " \\ " ) )
{
subkeyname = talloc_strdup ( frame , p2 ) ;
if ( ! subkeyname ) {
2008-04-13 13:38:44 +02:00
werr = WERR_NOMEM ;
2007-11-26 17:24:56 -08:00
goto fail ;
}
2006-12-03 17:34:11 +00:00
}
2008-03-20 13:59:09 +01:00
}
2006-12-03 17:34:11 +00:00
2008-03-20 13:59:09 +01:00
DEBUG ( 10 , ( " init_registry_key: Storing key [%s] with "
" subkey [%s] \n " , base ,
* subkeyname ? subkeyname : " NULL " ) ) ;
2007-11-26 17:24:56 -08:00
2008-03-20 13:59:09 +01:00
/* 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 */
2007-11-26 17:24:56 -08:00
2009-02-24 23:18:20 +01:00
werr = regsubkey_ctr_init ( frame , & subkeys ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
2008-03-20 13:59:09 +01:00
DEBUG ( 0 , ( " talloc() failure! \n " ) ) ;
goto fail ;
}
2006-12-03 17:34:11 +00:00
2009-07-15 12:45:43 +02:00
werr = regdb_fetch_keys_internal ( db , base , subkeys ) ;
if ( ! W_ERROR_IS_OK ( werr ) & &
! W_ERROR_EQUAL ( werr , WERR_NOT_FOUND ) )
{
goto fail ;
}
2008-03-20 13:59:09 +01:00
if ( * subkeyname ) {
2008-04-13 13:38:44 +02:00
werr = regsubkey_ctr_addkey ( subkeys , subkeyname ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
goto fail ;
}
2008-03-20 13:59:09 +01:00
}
2009-07-07 23:03:46 +02:00
if ( ! regdb_store_keys_internal ( db , base , subkeys ) ) {
2008-04-13 13:38:44 +02:00
werr = WERR_CAN_NOT_COMPLETE ;
2008-03-20 13:59:09 +01:00
goto fail ;
2005-05-23 16:25:31 +00:00
}
2008-03-20 13:59:09 +01:00
}
2008-04-13 13:38:44 +02:00
werr = WERR_OK ;
2008-03-20 13:59:09 +01:00
fail :
TALLOC_FREE ( frame ) ;
2008-04-13 13:38:44 +02:00
return werr ;
2008-03-20 13:59:09 +01:00
}
2007-12-13 22:20:58 +01:00
2009-07-07 23:58:03 +02:00
struct init_registry_key_context {
const char * add_path ;
} ;
static NTSTATUS init_registry_key_action ( struct db_context * db ,
void * private_data )
{
struct init_registry_key_context * init_ctx =
( struct init_registry_key_context * ) private_data ;
return werror_to_ntstatus ( init_registry_key_internal (
db , init_ctx - > add_path ) ) ;
}
2008-03-20 14:01:13 +01:00
/**
* Initialize a key in the registry :
* create each component key of the specified path ,
* wrapped in one db transaction .
*/
2008-04-13 13:38:44 +02:00
WERROR init_registry_key ( const char * add_path )
2008-03-20 14:01:13 +01:00
{
2009-07-07 23:58:03 +02:00
struct init_registry_key_context init_ctx ;
2008-04-13 13:38:44 +02:00
2009-07-03 17:39:17 +02:00
if ( regdb_key_exists ( regdb , add_path ) ) {
2008-04-29 17:18:26 +02:00
return WERR_OK ;
}
2009-07-07 23:58:03 +02:00
init_ctx . add_path = add_path ;
2008-03-20 14:01:13 +01:00
2009-07-07 23:58:03 +02:00
return ntstatus_to_werror ( dbwrap_trans_do ( regdb ,
init_registry_key_action ,
& init_ctx ) ) ;
2008-03-20 14:01:13 +01:00
}
2008-03-20 13:59:09 +01:00
/***********************************************************************
Open the registry data in the tdb
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-07-08 13:58:52 +02:00
static void regdb_ctr_add_value ( struct regval_ctr * ctr ,
struct builtin_regkey_value * value )
{
UNISTR2 data ;
switch ( value - > type ) {
case REG_DWORD :
regval_ctr_addvalue ( ctr , value - > valuename , REG_DWORD ,
( char * ) & value - > data . dw_value ,
sizeof ( uint32 ) ) ;
break ;
case REG_SZ :
init_unistr2 ( & data , value - > data . string , UNI_STR_TERMINATE ) ;
regval_ctr_addvalue ( ctr , value - > valuename , REG_SZ ,
( char * ) data . buffer ,
data . uni_str_len * sizeof ( uint16 ) ) ;
break ;
default :
DEBUG ( 0 , ( " regdb_ctr_add_value: invalid value type in "
" registry values [%d] \n " , value - > type ) ) ;
}
}
2009-07-08 13:10:37 +02:00
static NTSTATUS init_registry_data_action ( struct db_context * db ,
void * private_data )
2008-03-20 13:59:09 +01:00
{
2009-07-08 13:10:37 +02:00
NTSTATUS status ;
2008-04-29 17:37:35 +02:00
TALLOC_CTX * frame = talloc_stackframe ( ) ;
2009-03-23 18:14:17 +01:00
struct regval_ctr * values ;
2008-03-20 13:59:09 +01:00
int i ;
/* loop over all of the predefined paths and add each component */
for ( i = 0 ; builtin_registry_paths [ i ] ! = NULL ; i + + ) {
2009-07-08 13:10:37 +02:00
if ( regdb_key_exists ( db , builtin_registry_paths [ i ] ) ) {
2008-04-29 17:41:03 +02:00
continue ;
}
2009-07-08 13:10:37 +02:00
status = werror_to_ntstatus ( init_registry_key_internal ( db ,
builtin_registry_paths [ i ] ) ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto done ;
2008-03-20 13:59:09 +01:00
}
2005-05-23 16:25:31 +00:00
}
2005-05-09 13:51:44 +00:00
2005-06-25 17:31:40 +00:00
/* loop over all of the predefined values and add each component */
2007-11-26 17:24:56 -08:00
for ( i = 0 ; builtin_registry_values [ i ] . path ! = NULL ; i + + ) {
2007-12-13 22:20:58 +01:00
2009-03-23 18:14:17 +01:00
values = TALLOC_ZERO_P ( frame , struct regval_ctr ) ;
2008-03-17 12:21:32 +01:00
if ( values = = NULL ) {
2009-07-08 13:10:37 +02:00
status = NT_STATUS_NO_MEMORY ;
goto done ;
2005-08-29 14:55:40 +00:00
}
2009-07-08 13:10:37 +02:00
regdb_fetch_values_internal ( db ,
2009-07-08 12:32:48 +02:00
builtin_registry_values [ i ] . path ,
values ) ;
2006-12-03 17:34:11 +00:00
2008-03-17 00:02:52 +01:00
/* preserve existing values across restarts. Only add new ones */
2006-12-03 17:34:11 +00:00
2008-03-17 00:02:52 +01:00
if ( ! regval_ctr_key_exists ( values ,
builtin_registry_values [ i ] . valuename ) )
{
2009-07-08 13:58:52 +02:00
regdb_ctr_add_value ( values ,
& builtin_registry_values [ i ] ) ;
2009-07-08 13:10:37 +02:00
regdb_store_values_internal ( db ,
2009-07-08 12:38:41 +02:00
builtin_registry_values [ i ] . path ,
values ) ;
2005-06-25 17:31:40 +00:00
}
2008-03-17 00:02:52 +01:00
TALLOC_FREE ( values ) ;
2005-06-25 17:31:40 +00:00
}
2007-11-26 17:24:56 -08:00
2009-07-08 13:10:37 +02:00
status = NT_STATUS_OK ;
done :
TALLOC_FREE ( frame ) ;
return status ;
}
WERROR init_registry_data ( void )
{
WERROR werr ;
TALLOC_CTX * frame = talloc_stackframe ( ) ;
struct regval_ctr * values ;
int i ;
/*
* First , check for the existence of the needed keys and values .
* If all do already exist , we can save the writes .
*/
for ( i = 0 ; builtin_registry_paths [ i ] ! = NULL ; i + + ) {
if ( ! regdb_key_exists ( regdb , builtin_registry_paths [ i ] ) ) {
goto do_init ;
}
2007-02-07 13:26:13 +00:00
}
2009-07-08 13:10:37 +02:00
for ( i = 0 ; builtin_registry_values [ i ] . path ! = NULL ; i + + ) {
values = TALLOC_ZERO_P ( frame , struct regval_ctr ) ;
if ( values = = NULL ) {
werr = WERR_NOMEM ;
goto done ;
}
2007-12-13 22:20:58 +01:00
2009-07-08 13:10:37 +02:00
regdb_fetch_values_internal ( regdb ,
builtin_registry_values [ i ] . path ,
values ) ;
if ( ! regval_ctr_key_exists ( values ,
builtin_registry_values [ i ] . valuename ) )
{
TALLOC_FREE ( values ) ;
goto do_init ;
}
TALLOC_FREE ( values ) ;
2007-02-07 13:26:13 +00:00
}
2009-07-08 13:10:37 +02:00
werr = WERR_OK ;
goto done ;
do_init :
/*
* 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 .
*/
werr = ntstatus_to_werror ( dbwrap_trans_do ( regdb ,
init_registry_data_action ,
NULL ) ) ;
2008-04-29 17:37:35 +02:00
done :
TALLOC_FREE ( frame ) ;
2008-04-13 13:38:44 +02:00
return werr ;
2002-07-18 23:00:24 +00:00
}
/***********************************************************************
Open the registry database
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-04-13 12:41:34 +02:00
WERROR regdb_init ( void )
2002-07-18 23:00:24 +00:00
{
2005-06-17 15:53:01 +00:00
const char * vstring = " INFO/version " ;
uint32 vers_id ;
2008-04-13 12:41:34 +02:00
WERROR werr ;
2002-07-18 23:00:24 +00:00
2008-03-20 14:26:42 +01:00
if ( regdb ) {
DEBUG ( 10 , ( " regdb_init: incrementing refcount (%d) \n " ,
regdb_refcount ) ) ;
2008-03-16 23:08:33 +01:00
regdb_refcount + + ;
2008-04-13 12:41:34 +02:00
return WERR_OK ;
2008-02-19 01:04:31 +01:00
}
2002-07-18 23:00:24 +00:00
2008-08-07 16:20:05 +10:00
regdb = db_open ( NULL , state_path ( " registry.tdb " ) , 0 ,
2008-03-27 16:57:51 +01:00
REG_TDB_FLAGS , O_RDWR , 0600 ) ;
2008-03-20 14:24:12 +01:00
if ( ! regdb ) {
2008-08-07 16:20:05 +10:00
regdb = db_open ( NULL , state_path ( " registry.tdb " ) , 0 ,
2008-03-27 16:57:51 +01:00
REG_TDB_FLAGS , O_RDWR | O_CREAT , 0600 ) ;
2008-03-20 14:26:42 +01:00
if ( ! regdb ) {
2008-04-13 12:41:34 +02:00
werr = ntstatus_to_werror ( map_nt_error_from_unix ( errno ) ) ;
2008-07-15 14:22:55 +02:00
DEBUG ( 1 , ( " regdb_init: Failed to open registry %s (%s) \n " ,
2007-11-01 15:53:44 -04:00
state_path ( " registry.tdb " ) , strerror ( errno ) ) ) ;
2008-04-13 12:41:34 +02:00
return werr ;
2002-07-18 23:00:24 +00:00
}
2005-10-07 12:14:25 +00:00
DEBUG ( 10 , ( " regdb_init: Successfully created registry tdb \n " ) ) ;
2005-05-23 16:25:31 +00:00
}
2005-10-07 12:14:25 +00:00
2008-03-16 23:08:33 +01:00
regdb_refcount = 1 ;
2005-05-23 16:25:31 +00:00
2008-03-16 23:07:15 +01:00
vers_id = dbwrap_fetch_int32 ( regdb , vstring ) ;
2005-06-17 15:53:01 +00:00
if ( vers_id ! = REGVER_V1 ) {
2008-03-28 11:53:00 +01:00
NTSTATUS status ;
2005-09-01 14:00:53 +00:00
/* any upgrade code here if needed */
2008-03-20 14:35:41 +01:00
DEBUG ( 10 , ( " regdb_init: got %s = %d != %d \n " , vstring ,
2007-06-15 21:38:10 +00:00
vers_id , REGVER_V1 ) ) ;
2008-03-28 11:53:00 +01:00
status = dbwrap_trans_store_int32 ( regdb , vstring , REGVER_V1 ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2008-07-15 14:25:00 +02:00
DEBUG ( 1 , ( " regdb_init: error storing %s = %d: %s \n " ,
2008-03-28 11:53:00 +01:00
vstring , REGVER_V1 , nt_errstr ( status ) ) ) ;
2008-04-13 12:41:34 +02:00
return ntstatus_to_werror ( status ) ;
2008-03-20 14:33:43 +01:00
} else {
DEBUG ( 10 , ( " regdb_init: stored %s = %d \n " ,
vstring , REGVER_V1 ) ) ;
}
2005-09-01 14:00:53 +00:00
}
2005-06-17 15:53:01 +00:00
2008-04-13 12:41:34 +02:00
return WERR_OK ;
2002-07-18 23:00:24 +00:00
}
2005-10-07 12:14:25 +00:00
/***********************************************************************
Open the registry . Must already have been initialized by regdb_init ( )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
WERROR regdb_open ( void )
{
WERROR result = WERR_OK ;
2008-03-16 23:07:15 +01:00
if ( regdb ) {
2008-03-16 23:08:33 +01:00
DEBUG ( 10 , ( " regdb_open: incrementing refcount (%d) \n " , regdb_refcount ) ) ;
regdb_refcount + + ;
2005-10-07 12:14:25 +00:00
return WERR_OK ;
}
become_root ( ) ;
2008-08-07 16:20:05 +10:00
regdb = db_open ( NULL , state_path ( " registry.tdb " ) , 0 ,
2008-03-27 16:57:51 +01:00
REG_TDB_FLAGS , O_RDWR , 0600 ) ;
2008-03-16 23:07:15 +01:00
if ( ! regdb ) {
2005-10-07 12:14:25 +00:00
result = ntstatus_to_werror ( map_nt_error_from_unix ( errno ) ) ;
DEBUG ( 0 , ( " regdb_open: Failed to open %s! (%s) \n " ,
2007-11-01 15:53:44 -04:00
state_path ( " registry.tdb " ) , strerror ( errno ) ) ) ;
2005-10-07 12:14:25 +00:00
}
unbecome_root ( ) ;
2008-03-16 23:08:33 +01:00
regdb_refcount = 1 ;
DEBUG ( 10 , ( " regdb_open: refcount reset (%d) \n " , regdb_refcount ) ) ;
2005-10-07 12:14:25 +00:00
return result ;
}
/***********************************************************************
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
int regdb_close ( void )
{
2008-03-16 23:08:33 +01:00
if ( regdb_refcount = = 0 ) {
2008-01-04 12:57:49 +01:00
return 0 ;
}
2008-03-16 23:08:33 +01:00
regdb_refcount - - ;
2005-10-07 12:14:25 +00:00
2008-03-16 23:08:33 +01:00
DEBUG ( 10 , ( " regdb_close: decrementing refcount (%d) \n " , regdb_refcount ) ) ;
2005-10-07 12:14:25 +00:00
2008-03-16 23:08:33 +01:00
if ( regdb_refcount > 0 )
2005-10-07 12:14:25 +00:00
return 0 ;
2008-03-16 23:08:33 +01:00
SMB_ASSERT ( regdb_refcount > = 0 ) ;
2005-10-07 12:14:25 +00:00
2008-03-16 23:07:15 +01:00
TALLOC_FREE ( regdb ) ;
2007-06-15 10:40:36 +00:00
return 0 ;
2005-10-07 12:14:25 +00:00
}
2009-02-24 10:44:48 +01:00
WERROR regdb_transaction_start ( void )
{
return ( regdb - > transaction_start ( regdb ) = = 0 ) ?
WERR_OK : WERR_REG_IO_FAILURE ;
}
WERROR regdb_transaction_commit ( void )
{
return ( regdb - > transaction_commit ( regdb ) = = 0 ) ?
WERR_OK : WERR_REG_IO_FAILURE ;
}
WERROR regdb_transaction_cancel ( void )
{
return ( regdb - > transaction_cancel ( regdb ) = = 0 ) ?
WERR_OK : WERR_REG_IO_FAILURE ;
}
2007-06-13 13:15:16 +00: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 )
{
2008-03-16 23:07:15 +01:00
return regdb - > get_seqnum ( regdb ) ;
2007-06-13 13:15:16 +00:00
}
2009-02-26 01:16:07 +01:00
2009-07-03 16:40:43 +02:00
static WERROR regdb_delete_key_with_prefix ( struct db_context * db ,
const char * keyname ,
2009-02-26 01:16:07 +01:00
const char * prefix )
{
char * path ;
WERROR werr = WERR_NOMEM ;
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
if ( keyname = = NULL ) {
werr = WERR_INVALID_PARAM ;
goto done ;
}
if ( prefix = = NULL ) {
path = discard_const_p ( char , keyname ) ;
} else {
path = talloc_asprintf ( mem_ctx , " %s/%s " , prefix , keyname ) ;
if ( path = = NULL ) {
goto done ;
}
}
path = normalize_reg_path ( mem_ctx , path ) ;
if ( path = = NULL ) {
goto done ;
}
2009-07-03 16:40:43 +02:00
werr = ntstatus_to_werror ( dbwrap_delete_bystring ( db , path ) ) ;
2009-02-26 01:16:07 +01:00
/* treat "not" found" as ok */
if ( W_ERROR_EQUAL ( werr , WERR_NOT_FOUND ) ) {
werr = WERR_OK ;
}
done :
talloc_free ( mem_ctx ) ;
return werr ;
}
2009-07-03 16:42:20 +02:00
static WERROR regdb_delete_values ( struct db_context * db , const char * keyname )
2009-02-26 01:16:07 +01:00
{
2009-07-03 16:42:20 +02:00
return regdb_delete_key_with_prefix ( db , keyname , REG_VALUE_PREFIX ) ;
2009-02-26 01:16:07 +01:00
}
2009-07-03 16:44:20 +02:00
static WERROR regdb_delete_secdesc ( struct db_context * db , const char * keyname )
2009-02-26 01:16:07 +01:00
{
2009-07-03 16:44:20 +02:00
return regdb_delete_key_with_prefix ( db , keyname , REG_SECDESC_PREFIX ) ;
2009-02-26 01:16:07 +01:00
}
2009-07-03 16:48:36 +02:00
static WERROR regdb_delete_subkeylist ( struct db_context * db , const char * keyname )
2009-02-26 01:16:07 +01:00
{
2009-07-03 16:48:36 +02:00
return regdb_delete_key_with_prefix ( db , keyname , NULL ) ;
2009-02-26 01:16:07 +01:00
}
2009-07-03 16:51:26 +02:00
static WERROR regdb_delete_key_lists ( struct db_context * db , const char * keyname )
2009-02-26 01:43:58 +01:00
{
WERROR werr ;
2009-07-03 16:51:26 +02:00
werr = regdb_delete_values ( db , keyname ) ;
2009-02-26 01:43:58 +01:00
if ( ! W_ERROR_IS_OK ( werr ) ) {
DEBUG ( 1 , ( __location__ " Deleting %s/%s failed: %s \n " ,
REG_VALUE_PREFIX , keyname , win_errstr ( werr ) ) ) ;
goto done ;
}
2009-07-03 16:51:26 +02:00
werr = regdb_delete_secdesc ( db , keyname ) ;
2009-02-26 01:43:58 +01:00
if ( ! W_ERROR_IS_OK ( werr ) ) {
DEBUG ( 1 , ( __location__ " Deleting %s/%s failed: %s \n " ,
REG_SECDESC_PREFIX , keyname , win_errstr ( werr ) ) ) ;
goto done ;
}
2009-07-03 16:51:26 +02:00
werr = regdb_delete_subkeylist ( db , keyname ) ;
2009-02-26 01:43:58 +01:00
if ( ! W_ERROR_IS_OK ( werr ) ) {
DEBUG ( 1 , ( __location__ " Deleting %s failed: %s \n " ,
keyname , win_errstr ( werr ) ) ) ;
goto done ;
}
done :
return werr ;
}
2002-07-18 23:00:24 +00:00
/***********************************************************************
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-26 17:24:56 -08:00
2009-07-08 17:26:06 +02:00
static WERROR regdb_store_keys_internal2 ( struct db_context * db ,
const char * key ,
struct regsubkey_ctr * ctr )
2002-07-18 23:00:24 +00:00
{
2007-03-27 10:13:53 +00:00
TDB_DATA dbuf ;
2007-11-26 17:24:56 -08:00
uint8 * buffer = NULL ;
2002-07-18 23:00:24 +00:00
int i = 0 ;
uint32 len , buflen ;
2007-11-26 17:24:56 -08:00
uint32 num_subkeys = regsubkey_ctr_numkeys ( ctr ) ;
char * keyname = NULL ;
2008-03-17 00:09:07 +01:00
TALLOC_CTX * ctx = talloc_stackframe ( ) ;
2009-07-08 17:26:06 +02:00
WERROR werr ;
2005-05-23 16:25:31 +00:00
2007-11-26 17:24:56 -08:00
if ( ! key ) {
2009-07-09 00:21:46 +02:00
werr = WERR_INVALID_PARAM ;
goto done ;
2007-11-26 17:24:56 -08:00
}
keyname = talloc_strdup ( ctx , key ) ;
if ( ! keyname ) {
2009-07-09 00:21:46 +02:00
werr = WERR_NOMEM ;
goto done ;
2007-11-26 17:24:56 -08:00
}
2009-07-09 00:21:46 +02:00
2007-11-26 17:24:56 -08:00
keyname = normalize_reg_path ( ctx , keyname ) ;
2009-07-09 00:21:46 +02:00
if ( ! keyname ) {
werr = WERR_NOMEM ;
goto done ;
}
2005-06-17 18:57:37 +00:00
2002-07-18 23:00:24 +00:00
/* allocate some initial memory */
2007-11-26 17:24:56 -08:00
2008-03-17 00:22:12 +01:00
buffer = ( uint8 * ) SMB_MALLOC ( 1024 ) ;
if ( buffer = = NULL ) {
2009-07-08 17:26:06 +02:00
werr = WERR_NOMEM ;
goto done ;
2006-07-15 08:36:44 +00:00
}
2007-11-26 17:24:56 -08:00
buflen = 1024 ;
2002-07-18 23:00:24 +00:00
len = 0 ;
2007-11-26 17:24:56 -08:00
2002-07-18 23:00:24 +00:00
/* store the number of subkeys */
2007-11-26 17:24:56 -08:00
2008-03-17 00:22:12 +01:00
len + = tdb_pack ( buffer + len , buflen - len , " d " , num_subkeys ) ;
2007-11-26 17:24:56 -08:00
2002-07-18 23:00:24 +00:00
/* pack all the strings */
2007-11-26 17:24:56 -08:00
2002-07-18 23:00:24 +00:00
for ( i = 0 ; i < num_subkeys ; i + + ) {
2009-02-19 14:16:44 +01:00
size_t thistime ;
thistime = tdb_pack ( buffer + len , buflen - len , " f " ,
regsubkey_ctr_specific_key ( ctr , i ) ) ;
if ( len + thistime > buflen ) {
size_t thistime2 ;
/*
* tdb_pack hasn ' t done anything because of the short
* buffer , allocate extra space .
*/
buffer = SMB_REALLOC_ARRAY ( buffer , uint8_t ,
( len + thistime ) * 2 ) ;
2008-03-17 00:22:12 +01:00
if ( buffer = = NULL ) {
DEBUG ( 0 , ( " regdb_store_keys: Failed to realloc "
2009-02-19 13:11:36 -08:00
" memory of size [%u] \n " ,
( unsigned int ) ( len + thistime ) * 2 ) ) ;
2009-07-08 17:26:06 +02:00
werr = WERR_NOMEM ;
2009-02-19 14:16:44 +01:00
goto done ;
}
buflen = ( len + thistime ) * 2 ;
thistime2 = tdb_pack (
buffer + len , buflen - len , " f " ,
regsubkey_ctr_specific_key ( ctr , i ) ) ;
if ( thistime2 ! = thistime ) {
DEBUG ( 0 , ( " tdb_pack failed \n " ) ) ;
2009-07-08 17:26:06 +02:00
werr = WERR_CAN_NOT_COMPLETE ;
2002-07-18 23:00:24 +00:00
goto done ;
}
2007-11-26 17:24:56 -08:00
}
2009-02-19 14:16:44 +01:00
len + = thistime ;
2002-07-18 23:00:24 +00:00
}
2007-11-26 17:24:56 -08:00
2002-07-18 23:00:24 +00:00
/* finally write out the data */
2007-11-26 17:24:56 -08:00
2002-07-18 23:00:24 +00:00
dbuf . dptr = buffer ;
dbuf . dsize = len ;
2009-07-08 17:26:06 +02:00
werr = ntstatus_to_werror ( dbwrap_store_bystring ( db , keyname , dbuf ,
TDB_REPLACE ) ) ;
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
2002-07-18 23:00:24 +00:00
2009-02-20 06:01:16 +01:00
/*
* Delete a sorted subkey cache for regdb_key_exists , will be
* recreated automatically
*/
keyname = talloc_asprintf ( ctx , " %s/%s " , REG_SORTED_SUBKEYS_PREFIX ,
keyname ) ;
2009-07-08 17:26:06 +02:00
if ( keyname = = NULL ) {
werr = WERR_NOMEM ;
goto done ;
}
werr = ntstatus_to_werror ( dbwrap_delete_bystring ( db , keyname ) ) ;
/* don't treat WERR_NOT_FOUND as an error here */
if ( W_ERROR_EQUAL ( werr , WERR_NOT_FOUND ) ) {
werr = WERR_OK ;
2009-02-20 06:01:16 +01:00
}
2007-11-26 17:24:56 -08:00
done :
2008-03-17 00:09:07 +01:00
TALLOC_FREE ( ctx ) ;
2008-03-17 00:22:12 +01:00
SAFE_FREE ( buffer ) ;
2009-07-08 17:26:06 +02:00
return werr ;
2002-07-18 23:00:24 +00:00
}
2005-06-17 15:35:31 +00:00
/***********************************************************************
2007-11-26 17:24:56 -08:00
Store the new subkey record and create any child key records that
2005-06-17 15:35:31 +00:00
do not currently exist
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-07-09 10:28:29 +02:00
struct regdb_store_keys_context {
const char * key ;
struct regsubkey_ctr * ctr ;
} ;
static NTSTATUS regdb_store_keys_action ( struct db_context * db ,
void * private_data )
2005-06-17 15:35:31 +00:00
{
2009-07-09 10:28:29 +02:00
struct regdb_store_keys_context * store_ctx ;
WERROR werr ;
int num_subkeys , i ;
2007-11-26 17:24:56 -08:00
char * path = NULL ;
2009-02-24 15:19:18 +01:00
struct regsubkey_ctr * subkeys = NULL , * old_subkeys = NULL ;
2007-11-26 17:24:56 -08:00
char * oldkeyname = NULL ;
2009-07-09 10:28:29 +02:00
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
2008-03-17 00:43:56 +01:00
2009-07-09 10:28:29 +02:00
store_ctx = ( struct regdb_store_keys_context * ) private_data ;
2006-11-22 16:53:28 +00:00
2007-10-05 20:42:14 +00:00
/*
* Re - fetch the old keys inside the transaction
*/
2009-07-09 10:28:29 +02:00
werr = regsubkey_ctr_init ( mem_ctx , & old_subkeys ) ;
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
2005-08-29 14:55:40 +00:00
2009-07-15 12:45:43 +02:00
werr = regdb_fetch_keys_internal ( db , store_ctx - > key , old_subkeys ) ;
if ( ! W_ERROR_IS_OK ( werr ) & &
! W_ERROR_EQUAL ( werr , WERR_NOT_FOUND ) )
{
goto done ;
}
2007-11-26 17:24:56 -08:00
2008-05-06 10:06:34 +02:00
/*
* Make the store operation as safe as possible without transactions :
*
* ( 1 ) For each subkey removed from ctr compared with old_subkeys :
*
* ( a ) First delete the value db entry .
*
* ( b ) Next delete the secdesc db record .
*
* ( c ) Then delete the subkey list entry .
*
* ( 2 ) Now write the list of subkeys of the parent key ,
* deleting removed entries and adding new ones .
*
* ( 3 ) Finally create the subkey list entries for the added keys .
*
* This way if we crash half - way in between deleting the subkeys
* and storing the parent ' s list of subkeys , no old data can pop up
* out of the blue when re - adding keys later on .
*/
2008-05-08 16:55:14 +02:00
/* (1) delete removed keys' lists (values/secdesc/subkeys) */
2006-11-22 16:53:28 +00:00
2007-11-26 17:24:56 -08:00
num_subkeys = regsubkey_ctr_numkeys ( old_subkeys ) ;
for ( i = 0 ; i < num_subkeys ; i + + ) {
oldkeyname = regsubkey_ctr_specific_key ( old_subkeys , i ) ;
2009-07-09 10:28:29 +02:00
if ( regsubkey_ctr_key_exists ( store_ctx - > ctr , oldkeyname ) ) {
2006-11-22 16:53:28 +00:00
/*
* It ' s still around , don ' t delete
*/
continue ;
}
2009-07-09 10:28:29 +02:00
path = talloc_asprintf ( mem_ctx , " %s/%s " , store_ctx - > key ,
oldkeyname ) ;
2007-11-26 17:24:56 -08:00
if ( ! path ) {
2009-07-09 10:28:29 +02:00
werr = WERR_NOMEM ;
goto done ;
2007-11-26 17:24:56 -08:00
}
2009-02-26 01:16:07 +01:00
2009-07-07 11:31:28 +02:00
werr = regdb_delete_key_lists ( db , path ) ;
2009-07-09 10:28:29 +02:00
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
2009-02-26 01:16:07 +01:00
2007-11-26 17:24:56 -08:00
TALLOC_FREE ( path ) ;
2005-06-17 18:57:37 +00:00
}
2005-06-27 03:40:03 +00:00
2007-11-26 17:24:56 -08:00
TALLOC_FREE ( old_subkeys ) ;
2008-05-08 14:33:02 +02:00
/* (2) store the subkey list for the parent */
2009-07-09 10:28:29 +02:00
werr = regdb_store_keys_internal2 ( db , store_ctx - > key , store_ctx - > ctr ) ;
2009-07-08 17:26:06 +02:00
if ( ! W_ERROR_IS_OK ( werr ) ) {
2008-05-08 14:33:02 +02:00
DEBUG ( 0 , ( " regdb_store_keys: Failed to store new subkey list "
2009-07-09 10:28:29 +02:00
" for parent [%s]: %s \n " , store_ctx - > key ,
win_errstr ( werr ) ) ) ;
goto done ;
2008-05-08 14:33:02 +02:00
}
2008-05-08 16:32:51 +02:00
/* (3) now create records for any subkeys that don't already exist */
2005-08-29 14:55:40 +00:00
2009-07-09 10:28:29 +02:00
num_subkeys = regsubkey_ctr_numkeys ( store_ctx - > ctr ) ;
2008-03-01 00:32:36 +01:00
if ( num_subkeys = = 0 ) {
2009-07-09 10:28:29 +02:00
werr = regsubkey_ctr_init ( mem_ctx , & subkeys ) ;
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
2008-03-01 00:32:36 +01:00
2009-07-09 10:28:29 +02:00
werr = regdb_store_keys_internal2 ( db , store_ctx - > key , subkeys ) ;
2009-07-08 17:26:06 +02:00
if ( ! W_ERROR_IS_OK ( werr ) ) {
2008-03-01 00:32:36 +01:00
DEBUG ( 0 , ( " regdb_store_keys: Failed to store "
2009-07-09 10:28:29 +02:00
" new record for key [%s]: %s \n " ,
store_ctx - > key , win_errstr ( werr ) ) ) ;
goto done ;
2008-03-01 00:32:36 +01:00
}
TALLOC_FREE ( subkeys ) ;
}
2007-11-26 17:24:56 -08:00
for ( i = 0 ; i < num_subkeys ; i + + ) {
2009-07-09 10:28:29 +02:00
path = talloc_asprintf ( mem_ctx , " %s/%s " , store_ctx - > key ,
regsubkey_ctr_specific_key ( store_ctx - > ctr , i ) ) ;
2007-11-26 17:24:56 -08:00
if ( ! path ) {
2009-07-09 10:28:29 +02:00
werr = WERR_NOMEM ;
goto done ;
2005-08-29 14:55:40 +00:00
}
2009-07-09 10:28:29 +02:00
werr = regsubkey_ctr_init ( mem_ctx , & subkeys ) ;
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
2005-08-29 14:55:40 +00:00
2009-07-15 12:45:43 +02:00
werr = regdb_fetch_keys_internal ( db , path , subkeys ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
2005-06-17 15:35:31 +00:00
/* create a record with 0 subkeys */
2009-07-08 17:26:06 +02:00
werr = regdb_store_keys_internal2 ( db , path , subkeys ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
2006-11-22 15:10:46 +00:00
DEBUG ( 0 , ( " regdb_store_keys: Failed to store "
2009-07-08 17:26:06 +02:00
" new record for key [%s]: %s \n " , path ,
win_errstr ( werr ) ) ) ;
2009-07-09 10:28:29 +02:00
goto done ;
2005-06-17 15:35:31 +00:00
}
}
2005-08-29 14:55:40 +00:00
2007-11-26 17:24:56 -08:00
TALLOC_FREE ( subkeys ) ;
TALLOC_FREE ( path ) ;
2005-06-17 15:35:31 +00:00
}
2006-11-22 16:53:28 +00:00
2009-07-09 10:28:29 +02:00
werr = WERR_OK ;
done :
talloc_free ( mem_ctx ) ;
return werror_to_ntstatus ( werr ) ;
}
static bool regdb_store_keys_internal ( struct db_context * db , const char * key ,
struct regsubkey_ctr * ctr )
{
int num_subkeys , old_num_subkeys , i ;
struct regsubkey_ctr * old_subkeys = NULL ;
TALLOC_CTX * ctx = talloc_stackframe ( ) ;
WERROR werr ;
bool ret = false ;
struct regdb_store_keys_context store_ctx ;
if ( ! regdb_key_is_base_key ( key ) & & ! regdb_key_exists ( db , key ) ) {
2009-07-09 00:10:08 +02:00
goto done ;
2006-11-22 16:53:28 +00:00
}
2009-07-09 10:28:29 +02:00
/*
* fetch a list of the old subkeys so we can determine if anything has
* changed
*/
2006-11-22 16:53:28 +00:00
2009-07-09 10:28:29 +02:00
werr = regsubkey_ctr_init ( ctx , & old_subkeys ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
DEBUG ( 0 , ( " regdb_store_keys: talloc() failure! \n " ) ) ;
goto done ;
}
2009-07-15 12:45:43 +02:00
werr = regdb_fetch_keys_internal ( db , key , old_subkeys ) ;
if ( ! W_ERROR_IS_OK ( werr ) & &
! W_ERROR_EQUAL ( werr , WERR_NOT_FOUND ) )
{
goto done ;
}
2009-07-09 10:28:29 +02:00
num_subkeys = regsubkey_ctr_numkeys ( ctr ) ;
old_num_subkeys = regsubkey_ctr_numkeys ( old_subkeys ) ;
if ( ( num_subkeys & & old_num_subkeys ) & &
( num_subkeys = = old_num_subkeys ) ) {
for ( i = 0 ; i < num_subkeys ; i + + ) {
if ( strcmp ( regsubkey_ctr_specific_key ( ctr , i ) ,
regsubkey_ctr_specific_key ( old_subkeys , i ) )
! = 0 )
{
break ;
}
}
if ( i = = num_subkeys ) {
/*
* Nothing changed , no point to even start a tdb
* transaction
*/
ret = true ;
goto done ;
}
2006-11-22 16:53:28 +00:00
}
2009-07-09 10:28:29 +02:00
TALLOC_FREE ( old_subkeys ) ;
store_ctx . key = key ;
store_ctx . ctr = ctr ;
werr = ntstatus_to_werror ( dbwrap_trans_do ( db ,
regdb_store_keys_action ,
& store_ctx ) ) ;
ret = W_ERROR_IS_OK ( werr ) ;
2009-07-09 00:10:08 +02:00
done :
2008-03-17 01:00:40 +01:00
TALLOC_FREE ( ctx ) ;
2009-07-09 00:10:08 +02:00
return ret ;
2005-06-17 15:35:31 +00:00
}
2009-07-07 11:31:28 +02:00
bool regdb_store_keys ( const char * key , struct regsubkey_ctr * ctr )
{
return regdb_store_keys_internal ( regdb , key , ctr ) ;
}
2009-07-09 10:52:40 +02:00
/**
* create a subkey of a given key
*/
2009-07-09 10:41:59 +02:00
struct regdb_create_subkey_context {
const char * key ;
const char * subkey ;
} ;
static NTSTATUS regdb_create_subkey_action ( struct db_context * db ,
void * private_data )
2009-02-25 22:04:07 +01:00
{
WERROR werr ;
2009-07-09 10:41:59 +02:00
struct regdb_create_subkey_context * create_ctx ;
2009-02-25 22:04:07 +01:00
struct regsubkey_ctr * subkeys ;
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
2009-07-09 10:41:59 +02:00
create_ctx = ( struct regdb_create_subkey_context * ) private_data ;
2009-02-25 22:04:07 +01:00
werr = regsubkey_ctr_init ( mem_ctx , & subkeys ) ;
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
2009-07-15 12:45:43 +02:00
werr = regdb_fetch_keys_internal ( db , create_ctx - > key , subkeys ) ;
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
2009-02-25 22:04:07 +01:00
2009-07-09 10:41:59 +02:00
werr = regsubkey_ctr_addkey ( subkeys , create_ctx - > subkey ) ;
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
werr = regdb_store_keys_internal2 ( db , create_ctx - > key , subkeys ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
DEBUG ( 0 , ( __location__ " failed to store new subkey list for "
" parent key %s: %s \n " , create_ctx - > key ,
win_errstr ( werr ) ) ) ;
2009-02-25 22:04:07 +01:00
}
2009-07-09 10:41:59 +02:00
done :
talloc_free ( mem_ctx ) ;
return werror_to_ntstatus ( werr ) ;
}
2009-02-25 22:04:07 +01:00
2009-07-09 10:41:59 +02:00
static WERROR regdb_create_subkey ( const char * key , const char * subkey )
{
WERROR werr ;
struct regsubkey_ctr * subkeys ;
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
struct regdb_create_subkey_context create_ctx ;
if ( ! regdb_key_is_base_key ( key ) & & ! regdb_key_exists ( regdb , key ) ) {
werr = WERR_NOT_FOUND ;
2009-07-07 11:41:08 +02:00
goto done ;
}
2009-02-25 22:04:07 +01:00
werr = regsubkey_ctr_init ( mem_ctx , & subkeys ) ;
2009-07-09 10:41:59 +02:00
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
2009-02-25 22:04:07 +01:00
2009-07-15 12:45:43 +02:00
werr = regdb_fetch_keys_internal ( regdb , key , subkeys ) ;
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
2009-02-25 22:04:07 +01:00
2009-07-09 10:41:59 +02:00
if ( regsubkey_ctr_key_exists ( subkeys , subkey ) ) {
werr = WERR_OK ;
goto done ;
2009-02-25 22:04:07 +01:00
}
2009-07-09 10:41:59 +02:00
talloc_free ( subkeys ) ;
2009-02-25 22:04:07 +01:00
2009-07-09 10:41:59 +02:00
create_ctx . key = key ;
create_ctx . subkey = subkey ;
2009-02-25 22:04:07 +01:00
2009-07-09 10:41:59 +02:00
werr = ntstatus_to_werror ( dbwrap_trans_do ( regdb ,
regdb_create_subkey_action ,
& create_ctx ) ) ;
2009-02-25 22:04:07 +01:00
done :
talloc_free ( mem_ctx ) ;
return werr ;
}
2005-06-17 15:35:31 +00:00
2009-07-09 10:54:18 +02:00
/**
* create a subkey of a given key
*/
2009-07-09 11:04:20 +02:00
struct regdb_delete_subkey_context {
const char * key ;
const char * subkey ;
const char * path ;
} ;
static NTSTATUS regdb_delete_subkey_action ( struct db_context * db ,
void * private_data )
2009-02-26 02:56:00 +01:00
{
2009-07-07 12:27:26 +02:00
WERROR werr ;
2009-07-09 11:04:20 +02:00
struct regdb_delete_subkey_context * delete_ctx ;
2009-02-26 02:56:00 +01:00
struct regsubkey_ctr * subkeys ;
2009-07-09 11:04:20 +02:00
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
delete_ctx = ( struct regdb_delete_subkey_context * ) private_data ;
werr = regdb_delete_key_lists ( db , delete_ctx - > path ) ;
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
werr = regsubkey_ctr_init ( mem_ctx , & subkeys ) ;
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
2009-07-15 12:45:43 +02:00
werr = regdb_fetch_keys_internal ( db , delete_ctx - > key , subkeys ) ;
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
2009-07-09 11:04:20 +02:00
werr = regsubkey_ctr_delkey ( subkeys , delete_ctx - > subkey ) ;
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
werr = regdb_store_keys_internal2 ( db , delete_ctx - > key , subkeys ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
DEBUG ( 0 , ( __location__ " failed to store new subkey_list for "
" parent key %s: %s \n " , delete_ctx - > key ,
win_errstr ( werr ) ) ) ;
}
done :
talloc_free ( mem_ctx ) ;
return werror_to_ntstatus ( werr ) ;
}
static WERROR regdb_delete_subkey ( const char * key , const char * subkey )
{
WERROR werr ;
2009-02-26 02:56:00 +01:00
char * path ;
2009-07-09 11:04:20 +02:00
struct regdb_delete_subkey_context delete_ctx ;
2009-02-26 02:56:00 +01:00
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
2009-07-03 17:39:17 +02:00
if ( ! regdb_key_is_base_key ( key ) & & ! regdb_key_exists ( regdb , key ) ) {
2009-02-26 02:56:00 +01:00
werr = WERR_NOT_FOUND ;
goto done ;
}
path = talloc_asprintf ( mem_ctx , " %s/%s " , key , subkey ) ;
if ( path = = NULL ) {
werr = WERR_NOMEM ;
goto done ;
}
2009-07-03 17:39:17 +02:00
if ( ! regdb_key_exists ( regdb , path ) ) {
2009-02-26 02:56:00 +01:00
werr = WERR_OK ;
goto done ;
}
2009-07-09 11:04:20 +02:00
delete_ctx . key = key ;
delete_ctx . subkey = subkey ;
delete_ctx . path = path ;
2009-02-26 02:56:00 +01:00
2009-07-09 11:04:20 +02:00
werr = ntstatus_to_werror ( dbwrap_trans_do ( regdb ,
regdb_delete_subkey_action ,
& delete_ctx ) ) ;
2009-02-26 02:56:00 +01:00
done :
talloc_free ( mem_ctx ) ;
return werr ;
}
2009-07-03 17:10:09 +02:00
static TDB_DATA regdb_fetch_key_internal ( struct db_context * db ,
TALLOC_CTX * mem_ctx , const char * key )
2008-04-29 17:04:41 +02:00
{
char * path = NULL ;
2008-05-06 17:15:50 +02:00
TDB_DATA data ;
2008-04-29 17:04:41 +02:00
path = normalize_reg_path ( mem_ctx , key ) ;
if ( ! path ) {
return make_tdb_data ( NULL , 0 ) ;
}
2009-07-03 17:10:09 +02:00
data = dbwrap_fetch_bystring ( db , mem_ctx , path ) ;
2008-05-06 17:15:50 +02:00
TALLOC_FREE ( path ) ;
return data ;
2008-04-29 17:04:41 +02:00
}
2008-04-29 17:17:02 +02:00
2008-05-08 00:34:35 +02:00
/**
* check whether a given key name represents a base key ,
* i . e one without a subkey separator ( ' / ' or ' \ ' ) .
*/
static bool regdb_key_is_base_key ( const char * key )
{
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
bool ret = false ;
char * path ;
if ( key = = NULL ) {
goto done ;
}
path = normalize_reg_path ( mem_ctx , key ) ;
if ( path = = NULL ) {
DEBUG ( 0 , ( " out of memory! (talloc failed) \n " ) ) ;
goto done ;
}
if ( * path = = ' \0 ' ) {
goto done ;
}
ret = ( strrchr ( path , ' / ' ) = = NULL ) ;
done :
TALLOC_FREE ( mem_ctx ) ;
return ret ;
}
2009-02-22 01:11:51 +01:00
/*
* regdb_key_exists ( ) is a very frequent operation . It can be quite
* time - consuming to fully fetch the parent ' s subkey list , talloc_strdup all
* subkeys and then compare the keyname linearly to all the parent ' s subkeys .
*
* The following code tries to make this operation as efficient as possible :
* Per registry key we create a list of subkeys that is very efficient to
* search for existence of a subkey . Its format is :
*
* 4 bytes num_subkeys
* 4 * num_subkey bytes offset into the string array
* then follows a sorted list of subkeys in uppercase
*
* This record is created by create_sorted_subkeys ( ) on demand if it does not
* exist . scan_parent_subkeys ( ) uses regdb - > parse_record to search the sorted
* list , the parsing code and the binary search can be found in
* parent_subkey_scanner . The code uses parse_record ( ) to avoid a memcpy of
* the potentially large subkey record .
*
2009-07-07 11:11:10 +02:00
* The sorted subkey record is deleted in regdb_store_keys_internal2 and
2009-02-22 01:11:51 +01:00
* recreated on demand .
*/
2009-02-20 06:01:16 +01:00
static int cmp_keynames ( const void * p1 , const void * p2 )
{
return StrCaseCmp ( * ( ( char * * ) p1 ) , * ( ( char * * ) p2 ) ) ;
}
2009-07-13 17:15:14 +02:00
struct create_sorted_subkeys_context {
const char * key ;
const char * sorted_keyname ;
} ;
static NTSTATUS create_sorted_subkeys_action ( struct db_context * db ,
void * private_data )
2009-02-20 06:01:16 +01:00
{
char * * sorted_subkeys ;
2009-02-24 15:19:18 +01:00
struct regsubkey_ctr * ctr ;
2009-02-20 06:01:16 +01:00
NTSTATUS status ;
char * buf ;
char * p ;
2009-07-15 12:45:43 +02:00
int i ;
2009-02-20 06:01:16 +01:00
size_t len ;
2009-02-24 17:42:50 +01:00
int num_subkeys ;
2009-07-13 17:15:14 +02:00
struct create_sorted_subkeys_context * sorted_ctx ;
2009-02-20 06:01:16 +01:00
2009-07-13 17:15:14 +02:00
sorted_ctx = ( struct create_sorted_subkeys_context * ) private_data ;
2009-02-22 10:11:29 +01:00
2009-07-13 17:15:14 +02:00
/*
* In this function , we only treat failing of the actual write to
* the db as a real error . All preliminary errors , at a stage when
* nothing has been written to the DB yet are treated as success
* to be committed ( as an empty transaction ) .
*
* The reason is that this ( disposable ) call might be nested in other
* transactions . Doing a cancel here would destroy the possibility of
* a transaction_commit for transactions that we might be wrapped in .
*/
status = werror_to_ntstatus ( regsubkey_ctr_init ( talloc_tos ( ) , & ctr ) ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
/* don't treat this as an error */
status = NT_STATUS_OK ;
goto done ;
2009-02-20 06:01:16 +01:00
}
2009-07-15 12:45:43 +02:00
status = werror_to_ntstatus ( regdb_fetch_keys_internal ( db ,
sorted_ctx - > key ,
ctr ) ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-07-13 17:15:14 +02:00
/* don't treat this as an error */
2009-07-15 12:45:43 +02:00
status = NT_STATUS_OK ;
2009-07-13 17:15:14 +02:00
goto done ;
2009-02-20 06:01:16 +01:00
}
2009-02-24 17:42:50 +01:00
num_subkeys = regsubkey_ctr_numkeys ( ctr ) ;
sorted_subkeys = talloc_array ( ctr , char * , num_subkeys ) ;
2009-02-20 06:01:16 +01:00
if ( sorted_subkeys = = NULL ) {
2009-07-13 17:15:14 +02:00
/* don't treat this as an error */
goto done ;
2009-02-20 06:01:16 +01:00
}
2009-02-24 17:42:50 +01:00
len = 4 + 4 * num_subkeys ;
2009-02-20 06:01:16 +01:00
2009-02-24 17:42:50 +01:00
for ( i = 0 ; i < num_subkeys ; i + + ) {
2009-02-20 06:01:16 +01:00
sorted_subkeys [ i ] = talloc_strdup_upper ( sorted_subkeys ,
2009-02-24 22:16:40 +01:00
regsubkey_ctr_specific_key ( ctr , i ) ) ;
2009-02-20 06:01:16 +01:00
if ( sorted_subkeys [ i ] = = NULL ) {
2009-07-13 17:15:14 +02:00
/* don't treat this as an error */
goto done ;
2009-02-20 06:01:16 +01:00
}
len + = strlen ( sorted_subkeys [ i ] ) + 1 ;
}
2009-02-24 17:42:50 +01:00
qsort ( sorted_subkeys , num_subkeys , sizeof ( char * ) , cmp_keynames ) ;
2009-02-20 06:01:16 +01:00
buf = talloc_array ( ctr , char , len ) ;
if ( buf = = NULL ) {
2009-07-13 17:15:14 +02:00
/* don't treat this as an error */
goto done ;
2009-02-20 06:01:16 +01:00
}
2009-02-24 17:42:50 +01:00
p = buf + 4 + 4 * num_subkeys ;
2009-02-20 06:01:16 +01:00
2009-02-24 17:42:50 +01:00
SIVAL ( buf , 0 , num_subkeys ) ;
2009-02-20 06:01:16 +01:00
2009-02-24 17:42:50 +01:00
for ( i = 0 ; i < num_subkeys ; i + + ) {
2009-02-20 06:01:16 +01:00
ptrdiff_t offset = p - buf ;
SIVAL ( buf , 4 + 4 * i , offset ) ;
strlcpy ( p , sorted_subkeys [ i ] , len - offset ) ;
p + = strlen ( sorted_subkeys [ i ] ) + 1 ;
}
2009-02-22 10:11:29 +01:00
status = dbwrap_store_bystring (
2009-07-13 17:15:14 +02:00
db , sorted_ctx - > sorted_keyname , make_tdb_data ( ( uint8_t * ) buf ,
len ) ,
2009-02-20 06:01:16 +01:00
TDB_REPLACE ) ;
2009-07-13 17:15:14 +02:00
done :
talloc_free ( ctr ) ;
return status ;
}
2009-07-09 12:54:16 +02:00
2009-07-13 17:15:14 +02:00
static bool create_sorted_subkeys ( const char * key , const char * sorted_keyname )
{
NTSTATUS status ;
struct create_sorted_subkeys_context sorted_ctx ;
2009-07-09 12:54:16 +02:00
2009-07-13 17:15:14 +02:00
sorted_ctx . key = key ;
sorted_ctx . sorted_keyname = sorted_keyname ;
2009-02-22 10:11:29 +01:00
2009-07-13 17:15:14 +02:00
status = dbwrap_trans_do ( regdb ,
create_sorted_subkeys_action ,
& sorted_ctx ) ;
return NT_STATUS_IS_OK ( status ) ;
2009-02-20 06:01:16 +01:00
}
struct scan_subkey_state {
char * name ;
bool scanned ;
bool found ;
} ;
static int parent_subkey_scanner ( TDB_DATA key , TDB_DATA data ,
void * private_data )
{
struct scan_subkey_state * state =
( struct scan_subkey_state * ) private_data ;
uint32_t num_subkeys ;
uint32_t l , u ;
if ( data . dsize < sizeof ( uint32_t ) ) {
return - 1 ;
}
state - > scanned = true ;
state - > found = false ;
tdb_unpack ( data . dptr , data . dsize , " d " , & num_subkeys ) ;
l = 0 ;
u = num_subkeys ;
while ( l < u ) {
uint32_t idx = ( l + u ) / 2 ;
char * s = ( char * ) data . dptr + IVAL ( data . dptr , 4 + 4 * idx ) ;
int comparison = strcmp ( state - > name , s ) ;
if ( comparison < 0 ) {
u = idx ;
} else if ( comparison > 0 ) {
l = idx + 1 ;
} else {
state - > found = true ;
return 0 ;
}
}
return 0 ;
}
2009-07-03 17:18:19 +02:00
static bool scan_parent_subkeys ( struct db_context * db , const char * parent ,
const char * name )
2009-02-20 06:01:16 +01:00
{
char * path = NULL ;
char * key = NULL ;
struct scan_subkey_state state = { 0 , } ;
bool result = false ;
int res ;
state . name = NULL ;
path = normalize_reg_path ( talloc_tos ( ) , parent ) ;
if ( path = = NULL ) {
goto fail ;
}
key = talloc_asprintf ( talloc_tos ( ) , " %s/%s " ,
REG_SORTED_SUBKEYS_PREFIX , path ) ;
if ( key = = NULL ) {
goto fail ;
}
state . name = talloc_strdup_upper ( talloc_tos ( ) , name ) ;
if ( state . name = = NULL ) {
goto fail ;
}
state . scanned = false ;
2009-07-03 17:18:19 +02:00
res = db - > parse_record ( db , string_term_tdb_data ( key ) ,
parent_subkey_scanner , & state ) ;
2009-02-20 06:01:16 +01:00
if ( state . scanned ) {
result = state . found ;
} else {
if ( ! create_sorted_subkeys ( path , key ) ) {
goto fail ;
}
2009-07-03 17:18:19 +02:00
res = db - > parse_record ( db , string_term_tdb_data ( key ) ,
parent_subkey_scanner , & state ) ;
2009-02-20 06:01:16 +01:00
if ( ( res = = 0 ) & & ( state . scanned ) ) {
result = state . found ;
}
}
fail :
TALLOC_FREE ( path ) ;
TALLOC_FREE ( state . name ) ;
return result ;
}
2008-05-08 00:34:35 +02:00
2008-05-06 10:05:20 +02:00
/**
* Check for the existence of a key .
*
* Existence of a key is authoritatively defined by its
* existence in the list of subkeys of its parent key .
* The exeption of this are keys without a parent key ,
* i . e . the " base " keys ( HKLM , HKCU , . . . ) .
*/
2009-07-03 17:39:17 +02:00
static bool regdb_key_exists ( struct db_context * db , const char * key )
2008-04-29 17:17:02 +02:00
{
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
TDB_DATA value ;
2008-05-06 10:05:20 +02:00
bool ret = false ;
char * path , * p ;
if ( key = = NULL ) {
goto done ;
}
path = normalize_reg_path ( mem_ctx , key ) ;
if ( path = = NULL ) {
DEBUG ( 0 , ( " out of memory! (talloc failed) \n " ) ) ;
goto done ;
}
if ( * path = = ' \0 ' ) {
goto done ;
}
2008-04-29 17:17:02 +02:00
2008-05-06 10:05:20 +02:00
p = strrchr ( path , ' / ' ) ;
if ( p = = NULL ) {
/* this is a base key */
2009-07-03 17:39:17 +02:00
value = regdb_fetch_key_internal ( db , mem_ctx , path ) ;
2008-05-06 10:05:20 +02:00
ret = ( value . dptr ! = NULL ) ;
} else {
* p = ' \0 ' ;
2009-07-03 17:39:17 +02:00
ret = scan_parent_subkeys ( db , path , p + 1 ) ;
2008-05-06 10:05:20 +02:00
}
done :
2008-04-29 17:17:02 +02:00
TALLOC_FREE ( mem_ctx ) ;
return ret ;
}
2002-07-18 23:00:24 +00:00
/***********************************************************************
2007-11-26 17:24:56 -08:00
Retrieve an array of strings containing subkeys . Memory should be
released by the caller .
2002-07-18 23:00:24 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-07-15 12:45:43 +02:00
static WERROR regdb_fetch_keys_internal ( struct db_context * db , const char * key ,
struct regsubkey_ctr * ctr )
2002-07-18 23:00:24 +00:00
{
2009-02-24 17:51:09 +01:00
WERROR werr ;
2009-07-15 12:45:43 +02:00
uint32_t num_items ;
2007-03-29 09:35:51 +00:00
uint8 * buf ;
2002-07-18 23:00:24 +00:00
uint32 buflen , len ;
int i ;
2002-07-19 18:49:44 +00:00
fstring subkeyname ;
2007-12-13 22:20:58 +01:00
TALLOC_CTX * frame = talloc_stackframe ( ) ;
2008-04-01 12:38:54 +02:00
TDB_DATA value ;
2002-07-18 23:00:24 +00:00
2005-07-15 14:26:11 +00:00
DEBUG ( 11 , ( " regdb_fetch_keys: Enter key => [%s] \n " , key ? key : " NULL " ) ) ;
2007-11-26 17:24:56 -08:00
2009-07-15 12:45:43 +02:00
frame = talloc_stackframe ( ) ;
2009-07-03 17:39:17 +02:00
if ( ! regdb_key_exists ( db , key ) ) {
2009-07-15 12:45:43 +02:00
DEBUG ( 10 , ( " key [%s] not found \n " , key ) ) ;
werr = WERR_NOT_FOUND ;
2008-07-08 17:10:01 +02:00
goto done ;
2008-05-07 13:42:48 +02:00
}
2009-07-03 17:39:17 +02:00
werr = regsubkey_ctr_set_seqnum ( ctr , db - > get_seqnum ( db ) ) ;
2009-07-15 12:45:43 +02:00
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
2008-04-01 12:48:08 +02:00
2009-07-03 17:39:17 +02:00
value = regdb_fetch_key_internal ( db , frame , key ) ;
2008-01-14 18:31:11 +01:00
2008-07-08 17:10:01 +02:00
if ( value . dptr = = NULL ) {
DEBUG ( 10 , ( " regdb_fetch_keys: no subkeys found for key [%s] \n " ,
key ) ) ;
goto done ;
2002-07-18 23:00:24 +00:00
}
2007-11-26 17:24:56 -08:00
2008-07-08 17:10:01 +02:00
buf = value . dptr ;
buflen = value . dsize ;
2002-07-18 23:00:24 +00:00
len = tdb_unpack ( buf , buflen , " d " , & num_items ) ;
2007-11-26 17:24:56 -08:00
2009-07-15 12:50:55 +02:00
werr = regsubkey_ctr_reinit ( ctr ) ;
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
2002-07-18 23:00:24 +00:00
for ( i = 0 ; i < num_items ; i + + ) {
2007-11-26 17:24:56 -08:00
len + = tdb_unpack ( buf + len , buflen - len , " f " , subkeyname ) ;
2009-02-24 17:51:09 +01:00
werr = regsubkey_ctr_addkey ( ctr , subkeyname ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
DEBUG ( 5 , ( " regdb_fetch_keys: regsubkey_ctr_addkey "
" failed: %s \n " , win_errstr ( werr ) ) ) ;
2009-07-15 12:45:43 +02:00
num_items = 0 ;
2008-07-08 17:10:01 +02:00
goto done ;
2008-04-13 15:41:07 +02:00
}
2002-07-18 23:00:24 +00:00
}
2005-07-15 14:26:11 +00:00
DEBUG ( 11 , ( " regdb_fetch_keys: Exit [%d] items \n " , num_items ) ) ;
2007-11-26 17:24:56 -08:00
2008-07-08 17:10:01 +02:00
done :
2007-12-13 22:20:58 +01:00
TALLOC_FREE ( frame ) ;
2009-07-15 12:45:43 +02:00
return werr ;
2002-07-18 23:00:24 +00:00
}
2009-07-03 17:39:17 +02:00
int regdb_fetch_keys ( const char * key , struct regsubkey_ctr * ctr )
{
2009-07-15 12:45:43 +02:00
WERROR werr ;
werr = regdb_fetch_keys_internal ( regdb , key , ctr ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
return - 1 ;
}
return regsubkey_ctr_numkeys ( ctr ) ;
2009-07-03 17:39:17 +02:00
}
2005-06-25 17:31:40 +00:00
/****************************************************************************
Unpack a list of registry values frem the TDB
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-11-26 17:24:56 -08:00
2009-03-23 18:14:17 +01:00
static int regdb_unpack_values ( struct regval_ctr * values , uint8 * buf , int buflen )
2005-06-25 17:31:40 +00:00
{
int len = 0 ;
uint32 type ;
2007-11-26 17:24:56 -08:00
fstring valuename ;
2005-06-29 16:35:32 +00:00
uint32 size ;
2005-06-25 17:31:40 +00:00
uint8 * data_p ;
uint32 num_values = 0 ;
int i ;
2007-11-26 17:24:56 -08:00
2005-06-25 17:31:40 +00:00
/* loop and unpack the rest of the registry values */
2007-11-26 17:24:56 -08:00
2005-06-25 17:31:40 +00:00
len + = tdb_unpack ( buf + len , buflen - len , " d " , & num_values ) ;
2007-11-26 17:24:56 -08:00
2005-06-25 17:31:40 +00:00
for ( i = 0 ; i < num_values ; i + + ) {
/* unpack the next regval */
2007-11-26 17:24:56 -08:00
2005-06-29 16:35:32 +00:00
type = REG_NONE ;
size = 0 ;
data_p = NULL ;
2007-11-26 17:24:56 -08:00
valuename [ 0 ] = ' \0 ' ;
2005-06-25 17:31:40 +00:00
len + = tdb_unpack ( buf + len , buflen - len , " fdB " ,
valuename ,
& type ,
& size ,
& data_p ) ;
2007-11-26 17:24:56 -08:00
2005-06-29 16:35:32 +00:00
/* add the new value. Paranoid protective code -- make sure data_p is valid */
2005-06-25 17:31:40 +00:00
2007-11-26 17:24:56 -08:00
if ( * valuename & & size & & data_p ) {
regval_ctr_addvalue ( values , valuename , type ,
( const char * ) data_p , size ) ;
2005-06-29 16:35:32 +00:00
}
2007-11-26 17:24:56 -08:00
SAFE_FREE ( data_p ) ; /* 'B' option to tdb_unpack does a malloc() */
2005-06-25 17:31:40 +00:00
DEBUG ( 8 , ( " specific: [%s], len: %d \n " , valuename , size ) ) ;
}
return len ;
}
/****************************************************************************
Pack all values in all printer keys
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-11-26 17:24:56 -08:00
2009-03-23 18:14:17 +01:00
static int regdb_pack_values ( struct regval_ctr * values , uint8 * buf , int buflen )
2005-06-25 17:31:40 +00:00
{
int len = 0 ;
int i ;
2009-03-23 22:27:59 +01:00
struct regval_blob * val ;
2006-03-12 17:53:57 +00:00
int num_values ;
2005-06-25 17:31:40 +00:00
if ( ! values )
return 0 ;
2006-03-12 17:53:57 +00:00
num_values = regval_ctr_numvals ( values ) ;
2005-06-25 17:31:40 +00:00
/* pack the number of values first */
2007-11-26 17:24:56 -08:00
2005-06-25 17:31:40 +00:00
len + = tdb_pack ( buf + len , buflen - len , " d " , num_values ) ;
2007-11-26 17:24:56 -08:00
2005-06-25 17:31:40 +00:00
/* loop over all values */
2007-11-26 17:24:56 -08:00
for ( i = 0 ; i < num_values ; i + + ) {
2005-06-25 17:31:40 +00: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 ;
}
2002-07-19 18:49:44 +00:00
2002-07-18 23:00:24 +00:00
/***********************************************************************
2007-11-26 17:24:56 -08:00
Retrieve an array of strings containing subkeys . Memory should be
2005-06-25 17:31:40 +00:00
released by the caller .
2002-07-18 23:00:24 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-07-08 12:32:48 +02:00
static int regdb_fetch_values_internal ( struct db_context * db , const char * key ,
struct regval_ctr * values )
2002-07-18 23:00:24 +00:00
{
2007-11-26 17:24:56 -08:00
char * keystr = NULL ;
2008-03-16 23:03:34 +01:00
TALLOC_CTX * ctx = talloc_stackframe ( ) ;
int ret = 0 ;
2008-04-01 12:38:54 +02:00
TDB_DATA value ;
2005-05-09 13:51:44 +00:00
2005-06-30 02:59:29 +00:00
DEBUG ( 10 , ( " regdb_fetch_values: Looking for value of key [%s] \n " , key ) ) ;
2007-11-26 17:24:56 -08:00
2009-07-08 12:32:48 +02:00
if ( ! regdb_key_exists ( db , key ) ) {
2008-05-07 13:27:56 +02:00
goto done ;
}
2007-11-26 17:24:56 -08:00
keystr = talloc_asprintf ( ctx , " %s/%s " , REG_VALUE_PREFIX , key ) ;
if ( ! keystr ) {
2008-05-07 13:26:13 +02:00
goto done ;
2007-11-26 17:24:56 -08:00
}
2009-07-08 12:32:48 +02:00
values - > seqnum = db - > get_seqnum ( db ) ;
2008-04-01 12:48:08 +02:00
2009-07-08 12:32:48 +02:00
value = regdb_fetch_key_internal ( db , ctx , keystr ) ;
2008-01-14 18:31:11 +01:00
2008-04-01 12:38:54 +02:00
if ( ! value . dptr ) {
2005-06-27 03:40:03 +00:00
/* all keys have zero values by default */
2008-03-16 23:03:34 +01:00
goto done ;
2005-06-27 03:40:03 +00:00
}
2007-11-26 17:24:56 -08:00
2008-04-01 12:38:54 +02:00
regdb_unpack_values ( values , value . dptr , value . dsize ) ;
2008-03-16 23:03:34 +01:00
ret = regval_ctr_numvals ( values ) ;
2007-11-26 17:24:56 -08:00
2008-03-16 23:03:34 +01:00
done :
TALLOC_FREE ( ctx ) ;
return ret ;
2002-07-18 23:00:24 +00:00
}
2009-07-08 12:32:48 +02:00
int regdb_fetch_values ( const char * key , struct regval_ctr * values )
{
return regdb_fetch_values_internal ( regdb , key , values ) ;
}
2009-07-08 12:38:41 +02:00
static bool regdb_store_values_internal ( struct db_context * db , const char * key ,
struct regval_ctr * values )
2002-07-18 23:00:24 +00:00
{
2007-10-05 20:42:14 +00:00
TDB_DATA old_data , data ;
2007-11-26 17:24:56 -08:00
char * keystr = NULL ;
2008-03-17 01:20:18 +01:00
TALLOC_CTX * ctx = talloc_stackframe ( ) ;
2008-03-28 11:53:00 +01:00
int len ;
NTSTATUS status ;
2008-03-17 01:20:18 +01:00
bool result = false ;
2007-11-26 17:24:56 -08:00
2005-06-30 02:59:29 +00:00
DEBUG ( 10 , ( " regdb_store_values: Looking for value of key [%s] \n " , key ) ) ;
2007-11-26 17:24:56 -08:00
2009-07-08 12:38:41 +02:00
if ( ! regdb_key_exists ( db , key ) ) {
2008-05-07 13:45:02 +02:00
goto done ;
}
2007-11-26 17:24:56 -08:00
ZERO_STRUCT ( data ) ;
len = regdb_pack_values ( values , data . dptr , data . dsize ) ;
if ( len < = 0 ) {
2005-06-30 02:59:29 +00:00
DEBUG ( 0 , ( " regdb_store_values: unable to pack values. len <= 0 \n " ) ) ;
2008-03-17 01:20:18 +01:00
goto done ;
2005-06-25 17:31:40 +00:00
}
2007-11-26 17:24:56 -08:00
2008-03-17 01:26:35 +01:00
data . dptr = TALLOC_ARRAY ( ctx , uint8 , len ) ;
2005-06-25 17:31:40 +00:00
data . dsize = len ;
2007-11-26 17:24:56 -08:00
len = regdb_pack_values ( values , data . dptr , data . dsize ) ;
2005-06-25 17:31:40 +00:00
SMB_ASSERT ( len = = data . dsize ) ;
2007-11-26 17:24:56 -08:00
keystr = talloc_asprintf ( ctx , " %s/%s " , REG_VALUE_PREFIX , key ) ;
if ( ! keystr ) {
2008-03-17 01:20:18 +01:00
goto done ;
2007-11-26 17:24:56 -08:00
}
keystr = normalize_reg_path ( ctx , keystr ) ;
if ( ! keystr ) {
2008-03-17 01:20:18 +01:00
goto done ;
2007-11-26 17:24:56 -08:00
}
2009-07-08 12:38:41 +02:00
old_data = dbwrap_fetch_bystring ( db , ctx , keystr ) ;
2007-10-05 20:42:14 +00:00
if ( ( old_data . dptr ! = NULL )
& & ( old_data . dsize = = data . dsize )
2008-03-17 01:20:18 +01:00
& & ( memcmp ( old_data . dptr , data . dptr , data . dsize ) = = 0 ) )
{
result = true ;
goto done ;
2007-10-05 20:42:14 +00:00
}
2009-07-08 12:38:41 +02:00
status = dbwrap_trans_store_bystring ( db , keystr , data , TDB_REPLACE ) ;
2008-03-28 11:53:00 +01:00
result = NT_STATUS_IS_OK ( status ) ;
2007-11-26 17:24:56 -08:00
2008-03-17 01:20:18 +01:00
done :
TALLOC_FREE ( ctx ) ;
return result ;
2002-07-18 23:00:24 +00:00
}
2009-07-08 12:38:41 +02:00
bool regdb_store_values ( const char * key , struct regval_ctr * values )
{
return regdb_store_values_internal ( regdb , key , values ) ;
}
2006-11-30 07:38:40 +00:00
static WERROR regdb_get_secdesc ( TALLOC_CTX * mem_ctx , const char * key ,
struct security_descriptor * * psecdesc )
{
char * tdbkey ;
TDB_DATA data ;
NTSTATUS status ;
2008-03-16 23:03:34 +01:00
TALLOC_CTX * tmp_ctx = talloc_stackframe ( ) ;
2008-03-17 01:39:16 +01:00
WERROR err = WERR_OK ;
2006-11-30 07:38:40 +00:00
DEBUG ( 10 , ( " regdb_get_secdesc: Getting secdesc of key [%s] \n " , key ) ) ;
2009-07-03 17:39:17 +02:00
if ( ! regdb_key_exists ( regdb , key ) ) {
2008-05-07 13:48:28 +02:00
err = WERR_BADFILE ;
goto done ;
}
2008-03-17 01:39:16 +01:00
tdbkey = talloc_asprintf ( tmp_ctx , " %s/%s " , REG_SECDESC_PREFIX , key ) ;
if ( tdbkey = = NULL ) {
err = WERR_NOMEM ;
goto done ;
2006-11-30 07:38:40 +00:00
}
normalize_dbkey ( tdbkey ) ;
2008-03-16 23:07:15 +01:00
data = dbwrap_fetch_bystring ( regdb , tmp_ctx , tdbkey ) ;
2006-11-30 07:38:40 +00:00
if ( data . dptr = = NULL ) {
2008-03-17 01:39:16 +01:00
err = WERR_BADFILE ;
goto done ;
2006-11-30 07:38:40 +00:00
}
status = unmarshall_sec_desc ( mem_ctx , ( uint8 * ) data . dptr , data . dsize ,
psecdesc ) ;
if ( NT_STATUS_EQUAL ( status , NT_STATUS_NO_MEMORY ) ) {
2008-03-17 01:39:16 +01:00
err = WERR_NOMEM ;
} else if ( ! NT_STATUS_IS_OK ( status ) ) {
err = WERR_REG_CORRUPT ;
2006-11-30 07:38:40 +00:00
}
2008-03-17 01:39:16 +01:00
done :
TALLOC_FREE ( tmp_ctx ) ;
return err ;
2006-11-30 07:38:40 +00:00
}
static WERROR regdb_set_secdesc ( const char * key ,
struct security_descriptor * secdesc )
{
2008-03-17 01:27:27 +01:00
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
2006-11-30 07:38:40 +00:00
char * tdbkey ;
WERROR err = WERR_NOMEM ;
TDB_DATA tdbdata ;
2008-03-17 01:31:47 +01:00
2009-07-03 17:39:17 +02:00
if ( ! regdb_key_exists ( regdb , key ) ) {
2008-05-07 13:50:01 +02:00
err = WERR_BADFILE ;
goto done ;
}
2008-03-17 01:31:47 +01:00
tdbkey = talloc_asprintf ( mem_ctx , " %s/%s " , REG_SECDESC_PREFIX , key ) ;
if ( tdbkey = = NULL ) {
2006-11-30 07:38:40 +00:00
goto done ;
}
normalize_dbkey ( tdbkey ) ;
2007-11-06 00:50:47 +01:00
if ( secdesc = = NULL ) {
/* assuming a delete */
2009-02-26 01:22:03 +01:00
err = ntstatus_to_werror ( dbwrap_trans_delete_bystring ( regdb ,
tdbkey ) ) ;
2007-11-06 00:50:47 +01:00
goto done ;
}
2007-03-27 10:05:20 +00:00
err = ntstatus_to_werror ( marshall_sec_desc ( mem_ctx , secdesc ,
2007-06-13 10:07:05 +00:00
& tdbdata . dptr ,
2006-11-30 07:38:40 +00:00
& tdbdata . dsize ) ) ;
2009-02-26 01:22:03 +01:00
W_ERROR_NOT_OK_GOTO_DONE ( err ) ;
2006-11-30 07:38:40 +00:00
2009-02-26 01:22:03 +01:00
err = ntstatus_to_werror ( dbwrap_trans_store_bystring ( regdb , tdbkey ,
tdbdata , 0 ) ) ;
2006-11-30 07:38:40 +00:00
done :
TALLOC_FREE ( mem_ctx ) ;
return err ;
}
2002-07-19 18:49:44 +00:00
2009-02-24 15:19:18 +01:00
bool regdb_subkeys_need_update ( struct regsubkey_ctr * subkeys )
2008-01-14 18:31:11 +01:00
{
2009-02-24 23:19:35 +01:00
return ( regdb_get_seqnum ( ) ! = regsubkey_ctr_get_seqnum ( subkeys ) ) ;
2008-01-14 18:31:11 +01:00
}
2009-03-23 18:14:17 +01:00
bool regdb_values_need_update ( struct regval_ctr * values )
2008-01-14 18:31:11 +01:00
{
return ( regdb_get_seqnum ( ) ! = values - > seqnum ) ;
}
2002-07-19 18:49:44 +00:00
/*
* Table of function pointers for default access
*/
2009-03-23 23:14:45 +01:00
struct registry_ops regdb_ops = {
2008-03-17 01:44:26 +01:00
. fetch_subkeys = regdb_fetch_keys ,
. fetch_values = regdb_fetch_values ,
. store_subkeys = regdb_store_keys ,
. store_values = regdb_store_values ,
2009-02-25 22:04:07 +01:00
. create_subkey = regdb_create_subkey ,
2009-02-26 02:56:00 +01:00
. delete_subkey = regdb_delete_subkey ,
2008-03-17 01:44:26 +01:00
. get_secdesc = regdb_get_secdesc ,
. set_secdesc = regdb_set_secdesc ,
. subkeys_need_update = regdb_subkeys_need_update ,
. values_need_update = regdb_values_need_update
2002-07-19 18:49:44 +00:00
} ;