2002-07-16 02:27:07 +04:00
/*
* Unix SMB / CIFS implementation .
2005-05-23 20:25:31 +04:00
* Virtual Windows Registry Layer
* Copyright ( C ) Gerald Carter 2002 - 2005
2002-07-16 02:27:07 +04:00
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
2002-07-19 03:00:24 +04:00
/* Implementation of registry frontend view functions. */
2002-07-16 02:27:07 +04:00
# include "includes.h"
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_RPC_SRV
2002-07-19 03:00:24 +04:00
extern REGISTRY_OPS printing_ops ;
2005-03-24 02:26:33 +03:00
extern REGISTRY_OPS eventlog_ops ;
2005-05-23 20:25:31 +04:00
extern REGISTRY_OPS shares_reg_ops ;
2002-07-19 22:49:44 +04:00
extern REGISTRY_OPS regdb_ops ; /* these are the default */
2002-07-16 02:27:07 +04:00
2002-07-19 03:00:24 +04:00
/* array of REGISTRY_HOOK's which are read into a tree for easy access */
2002-07-16 02:27:07 +04:00
2002-07-19 03:00:24 +04:00
REGISTRY_HOOK reg_hooks [ ] = {
2005-05-23 20:25:31 +04:00
{ KEY_PRINTING , & printing_ops } ,
{ KEY_PRINTING_2K , & printing_ops } ,
{ KEY_PRINTING_PORTS , & printing_ops } ,
{ KEY_EVENTLOG , & eventlog_ops } ,
{ KEY_SHARES , & shares_reg_ops } ,
2002-07-19 03:00:24 +04:00
{ NULL , NULL }
} ;
2002-07-16 02:27:07 +04:00
2002-07-23 08:55:06 +04:00
/***********************************************************************
Open the registry database and initialize the REGISTRY_HOOK cache
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
BOOL init_registry ( void )
{
int i ;
if ( ! init_registry_db ( ) ) {
DEBUG ( 0 , ( " init_registry: failed to initialize the registry tdb! \n " ) ) ;
return False ;
}
/* build the cache tree of registry hooks */
reghook_cache_init ( ) ;
for ( i = 0 ; reg_hooks [ i ] . keyname ; i + + ) {
if ( ! reghook_cache_add ( & reg_hooks [ i ] ) )
return False ;
}
if ( DEBUGLEVEL > = 20 )
reghook_dump_cache ( 20 ) ;
return True ;
}
/***********************************************************************
High level wrapper function for storing registry subkeys
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
BOOL store_reg_keys ( REGISTRY_KEY * key , REGSUBKEY_CTR * subkeys )
{
2005-06-17 00:45:55 +04:00
if ( key - > hook & & key - > hook - > ops & & key - > hook - > ops - > store_subkeys )
return key - > hook - > ops - > store_subkeys ( key - > name , subkeys ) ;
2005-06-25 21:31:40 +04:00
return False ;
2002-07-23 08:55:06 +04:00
}
/***********************************************************************
High level wrapper function for storing registry values
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
BOOL store_reg_values ( REGISTRY_KEY * key , REGVAL_CTR * val )
{
2005-06-27 07:40:03 +04:00
if ( check_dynamic_reg_values ( key ) )
return False ;
2005-06-17 00:45:55 +04:00
if ( key - > hook & & key - > hook - > ops & & key - > hook - > ops - > store_values )
return key - > hook - > ops - > store_values ( key - > name , val ) ;
2005-06-25 21:31:40 +04:00
return False ;
2002-07-23 08:55:06 +04:00
}
/***********************************************************************
High level wrapper function for enumerating registry subkeys
Initialize the TALLOC_CTX if necessary
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
int fetch_reg_keys ( REGISTRY_KEY * key , REGSUBKEY_CTR * subkey_ctr )
{
int result = - 1 ;
2005-06-17 00:45:55 +04:00
if ( key - > hook & & key - > hook - > ops & & key - > hook - > ops - > fetch_subkeys )
result = key - > hook - > ops - > fetch_subkeys ( key - > name , subkey_ctr ) ;
2002-07-23 08:55:06 +04:00
return result ;
}
/***********************************************************************
retreive a specific subkey specified by index . Caller is
responsible for freeing memory
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
BOOL fetch_reg_keys_specific ( REGISTRY_KEY * key , char * * subkey , uint32 key_index )
{
2002-07-24 10:42:09 +04:00
static REGSUBKEY_CTR ctr ;
static pstring save_path ;
static BOOL ctr_init = False ;
2002-07-23 08:55:06 +04:00
char * s ;
2002-07-24 10:42:09 +04:00
* subkey = NULL ;
2002-07-23 08:55:06 +04:00
2002-07-24 10:42:09 +04:00
/* simple caching for performance; very basic heuristic */
2005-03-24 02:26:33 +03:00
DEBUG ( 8 , ( " fetch_reg_keys_specific: Looking for key [%d] of [%s] \n " , key_index , key - > name ) ) ;
2002-07-23 08:55:06 +04:00
2002-07-24 10:42:09 +04:00
if ( ! ctr_init ) {
DEBUG ( 8 , ( " fetch_reg_keys_specific: Initializing cache of subkeys for [%s] \n " , key - > name ) ) ;
regsubkey_ctr_init ( & ctr ) ;
pstrcpy ( save_path , key - > name ) ;
if ( fetch_reg_keys ( key , & ctr ) = = - 1 )
return False ;
ctr_init = True ;
}
/* clear the cache when key_index == 0 or the path has changed */
else if ( ! key_index | | StrCaseCmp ( save_path , key - > name ) ) {
2002-07-23 08:55:06 +04:00
2002-07-24 10:42:09 +04:00
DEBUG ( 8 , ( " fetch_reg_keys_specific: Updating cache of subkeys for [%s] \n " , key - > name ) ) ;
regsubkey_ctr_destroy ( & ctr ) ;
regsubkey_ctr_init ( & ctr ) ;
pstrcpy ( save_path , key - > name ) ;
if ( fetch_reg_keys ( key , & ctr ) = = - 1 )
return False ;
}
2002-07-23 08:55:06 +04:00
if ( ! ( s = regsubkey_ctr_specific_key ( & ctr , key_index ) ) )
return False ;
2004-12-07 21:25:53 +03:00
* subkey = SMB_STRDUP ( s ) ;
2002-07-23 08:55:06 +04:00
return True ;
}
/***********************************************************************
High level wrapper function for enumerating registry values
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
int fetch_reg_values ( REGISTRY_KEY * key , REGVAL_CTR * val )
{
int result = - 1 ;
2005-06-17 00:45:55 +04:00
if ( key - > hook & & key - > hook - > ops & & key - > hook - > ops - > fetch_values )
result = key - > hook - > ops - > fetch_values ( key - > name , val ) ;
2005-06-27 07:40:03 +04:00
/* if the backend lookup returned no data, try the dynamic overlay */
if ( result = = 0 ) {
result = fetch_dynamic_reg_values ( key , val ) ;
2002-07-23 08:55:06 +04:00
2005-06-27 07:40:03 +04:00
return ( result ! = - 1 ) ? result : 0 ;
}
2002-07-23 08:55:06 +04:00
return result ;
}
/***********************************************************************
retreive a specific subkey specified by index . Caller is
responsible for freeing memory
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2002-07-24 10:42:09 +04:00
BOOL fetch_reg_values_specific ( REGISTRY_KEY * key , REGISTRY_VALUE * * val , uint32 val_index )
2002-07-23 08:55:06 +04:00
{
2002-07-24 10:42:09 +04:00
static REGVAL_CTR ctr ;
static pstring save_path ;
static BOOL ctr_init = False ;
REGISTRY_VALUE * v ;
2002-07-23 08:55:06 +04:00
2002-07-24 10:42:09 +04:00
* val = NULL ;
2002-07-23 08:55:06 +04:00
2002-07-24 10:42:09 +04:00
/* simple caching for performance; very basic heuristic */
2002-07-23 08:55:06 +04:00
2002-07-24 10:42:09 +04:00
if ( ! ctr_init ) {
DEBUG ( 8 , ( " fetch_reg_values_specific: Initializing cache of values for [%s] \n " , key - > name ) ) ;
2002-07-23 08:55:06 +04:00
2002-07-24 10:42:09 +04:00
regval_ctr_init ( & ctr ) ;
pstrcpy ( save_path , key - > name ) ;
if ( fetch_reg_values ( key , & ctr ) = = - 1 )
return False ;
ctr_init = True ;
}
/* clear the cache when val_index == 0 or the path has changed */
2005-06-27 07:40:03 +04:00
else if ( ! val_index | | ! strequal ( save_path , key - > name ) ) {
2002-07-24 10:42:09 +04:00
DEBUG ( 8 , ( " fetch_reg_values_specific: Updating cache of values for [%s] \n " , key - > name ) ) ;
regval_ctr_destroy ( & ctr ) ;
regval_ctr_init ( & ctr ) ;
pstrcpy ( save_path , key - > name ) ;
if ( fetch_reg_values ( key , & ctr ) = = - 1 )
return False ;
}
if ( ! ( v = regval_ctr_specific_value ( & ctr , val_index ) ) )
2002-07-23 08:55:06 +04:00
return False ;
* val = dup_registry_value ( v ) ;
return True ;
}
2005-06-17 05:57:18 +04:00
/***********************************************************************
High level access check for passing the required access mask to the
underlying registry backend
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
BOOL regkey_access_check ( REGISTRY_KEY * key , uint32 requested , uint32 * granted , NT_USER_TOKEN * token )
{
/* use the default security check if the backend has not defined its own */
if ( ! ( key - > hook & & key - > hook - > ops & & key - > hook - > ops - > reg_access_check ) ) {
SEC_DESC * sec_desc ;
NTSTATUS status ;
if ( ! ( sec_desc = construct_registry_sd ( get_talloc_ctx ( ) ) ) )
return False ;
status = registry_access_check ( sec_desc , token , requested , granted ) ;
return NT_STATUS_IS_OK ( status ) ;
}
return key - > hook - > ops - > reg_access_check ( key - > name , requested , granted , token ) ;
}
2002-07-23 08:55:06 +04:00