2008-01-15 14:55:04 +01:00
/*
2006-12-01 20:01:09 +00:00
* Unix SMB / CIFS implementation .
* Virtual Windows Registry Layer
* Copyright ( C ) Volker Lendecke 2006
2010-09-21 07:44:56 +02:00
* Copyright ( C ) Michael Adam 2007 - 2010
2006-12-01 20:01:09 +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
2006-12-01 20:01:09 +00:00
* ( at your option ) any later version .
2008-01-15 14:55:04 +01:00
*
2006-12-01 20:01:09 +00:00
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
2008-01-15 14:55:04 +01:00
*
2006-12-01 20:01:09 +00:00
* 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/>.
2006-12-01 20:01:09 +00:00
*/
/* Attempt to wrap the existing API in a more winreg.idl-like way */
2008-01-17 11:07:28 +01:00
/*
* Here is a list of winreg . idl functions and corresponding implementations
* provided here :
*
* 0x00 winreg_OpenHKCR
* 0x01 winreg_OpenHKCU
* 0x02 winreg_OpenHKLM
* 0x03 winreg_OpenHKPD
* 0x04 winreg_OpenHKU
* 0x05 winreg_CloseKey
* 0x06 winreg_CreateKey reg_createkey
* 0x07 winreg_DeleteKey reg_deletekey
* 0x08 winreg_DeleteValue reg_deletevalue
* 0x09 winreg_EnumKey reg_enumkey
* 0x0a winreg_EnumValue reg_enumvalue
* 0x0b winreg_FlushKey
* 0x0c winreg_GetKeySecurity reg_getkeysecurity
* 0x0d winreg_LoadKey
* 0x0e winreg_NotifyChangeKeyValue
* 0x0f winreg_OpenKey reg_openkey
* 0x10 winreg_QueryInfoKey reg_queryinfokey
* 0x11 winreg_QueryValue reg_queryvalue
* 0x12 winreg_ReplaceKey
2008-02-15 15:31:31 +01:00
* 0x13 winreg_RestoreKey reg_restorekey
2008-02-15 15:14:20 +01:00
* 0x14 winreg_SaveKey reg_savekey
2008-01-17 11:07:28 +01:00
* 0x15 winreg_SetKeySecurity reg_setkeysecurity
* 0x16 winreg_SetValue reg_setvalue
* 0x17 winreg_UnLoadKey
* 0x18 winreg_InitiateSystemShutdown
* 0x19 winreg_AbortSystemShutdown
2008-01-17 11:22:01 +01:00
* 0x1a winreg_GetVersion reg_getversion
2008-01-17 11:07:28 +01:00
* 0x1b winreg_OpenHKCC
* 0x1c winreg_OpenHKDD
2010-09-20 01:51:54 +02:00
* 0x1d winreg_QueryMultipleValues reg_querymultiplevalues
2008-01-17 11:07:28 +01:00
* 0x1e winreg_InitiateSystemShutdownEx
* 0x1f winreg_SaveKeyEx
* 0x20 winreg_OpenHKPT
* 0x21 winreg_OpenHKPN
2010-09-20 01:51:54 +02:00
* 0x22 winreg_QueryMultipleValues2 reg_querymultiplevalues
2008-01-17 11:07:28 +01:00
*
*/
2006-12-01 20:01:09 +00:00
# include "includes.h"
2009-10-02 00:17:06 +02:00
# include "registry.h"
2010-09-21 08:11:18 +02:00
# include "reg_api.h"
2010-05-23 15:25:00 +02:00
# include "reg_cachehook.h"
2010-05-24 22:42:00 +02:00
# include "reg_backend_db.h"
2010-05-24 22:49:42 +02:00
# include "reg_dispatcher.h"
2010-05-25 01:00:37 +02:00
# include "reg_objects.h"
2010-05-28 02:19:32 +02:00
# include "../librpc/gen_ndr/ndr_security.h"
2006-12-01 20:01:09 +00:00
2007-09-28 23:05:52 +00:00
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_REGISTRY
2008-01-17 00:57:53 +01:00
/**********************************************************************
* Helper functions
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-12-01 20:01:09 +00:00
static WERROR fill_value_cache ( struct registry_key * key )
{
2010-05-23 23:56:40 +02:00
WERROR werr ;
2006-12-01 20:01:09 +00:00
if ( key - > values ! = NULL ) {
2008-01-14 18:31:11 +01:00
if ( ! reg_values_need_update ( key - > key , key - > values ) ) {
return WERR_OK ;
}
2006-12-01 20:01:09 +00:00
}
2010-05-23 23:56:40 +02:00
werr = regval_ctr_init ( key , & ( key - > values ) ) ;
W_ERROR_NOT_OK_RETURN ( werr ) ;
2006-12-01 20:01:09 +00:00
if ( fetch_reg_values ( key - > key , key - > values ) = = - 1 ) {
TALLOC_FREE ( key - > values ) ;
return WERR_BADFILE ;
}
return WERR_OK ;
}
static WERROR fill_subkey_cache ( struct registry_key * key )
{
2009-02-24 23:17:05 +01:00
WERROR werr ;
2006-12-01 20:01:09 +00:00
if ( key - > subkeys ! = NULL ) {
2008-01-14 18:31:11 +01:00
if ( ! reg_subkeys_need_update ( key - > key , key - > subkeys ) ) {
return WERR_OK ;
}
2006-12-01 20:01:09 +00:00
}
2011-07-14 12:19:01 +02:00
TALLOC_FREE ( key - > subkeys ) ;
2009-02-24 23:17:05 +01:00
werr = regsubkey_ctr_init ( key , & ( key - > subkeys ) ) ;
W_ERROR_NOT_OK_RETURN ( werr ) ;
2006-12-01 20:01:09 +00:00
if ( fetch_reg_keys ( key - > key , key - > subkeys ) = = - 1 ) {
TALLOC_FREE ( key - > subkeys ) ;
return WERR_NO_MORE_ITEMS ;
}
return WERR_OK ;
}
2009-03-23 23:02:57 +01:00
static int regkey_destructor ( struct registry_key_handle * key )
2007-06-22 11:03:48 +00:00
{
return regdb_close ( ) ;
}
static WERROR regkey_open_onelevel ( TALLOC_CTX * mem_ctx ,
struct registry_key * parent ,
const char * name ,
2010-08-26 20:04:11 +10:00
const struct security_token * token ,
2007-06-22 11:03:48 +00:00
uint32 access_desired ,
struct registry_key * * pregkey )
{
WERROR result = WERR_OK ;
struct registry_key * regkey ;
2009-03-23 23:02:57 +01:00
struct registry_key_handle * key ;
2009-02-24 15:19:18 +01:00
struct regsubkey_ctr * subkeys = NULL ;
2007-06-22 11:03:48 +00:00
DEBUG ( 7 , ( " regkey_open_onelevel: name = [%s] \n " , name ) ) ;
SMB_ASSERT ( strchr ( name , ' \\ ' ) = = NULL ) ;
2011-06-07 11:44:43 +10:00
if ( ! ( regkey = talloc_zero ( mem_ctx , struct registry_key ) ) | |
2007-06-22 11:03:48 +00:00
! ( regkey - > token = dup_nt_token ( regkey , token ) ) | |
2011-06-07 11:44:43 +10:00
! ( regkey - > key = talloc_zero ( regkey , struct registry_key_handle ) ) )
2009-03-23 23:02:57 +01:00
{
2007-06-22 11:03:48 +00:00
result = WERR_NOMEM ;
goto done ;
}
if ( ! ( W_ERROR_IS_OK ( result = regdb_open ( ) ) ) ) {
goto done ;
}
key = regkey - > key ;
talloc_set_destructor ( key , regkey_destructor ) ;
2010-02-07 11:05:07 +01:00
2007-06-22 11:03:48 +00:00
/* initialization */
2010-02-07 11:05:07 +01:00
2007-06-22 11:03:48 +00:00
key - > type = REG_KEY_GENERIC ;
if ( name [ 0 ] = = ' \0 ' ) {
/*
* Open a copy of the parent key
*/
if ( ! parent ) {
result = WERR_BADFILE ;
goto done ;
}
key - > name = talloc_strdup ( key , parent - > key - > name ) ;
}
else {
/*
* Normal subkey open
*/
key - > name = talloc_asprintf ( key , " %s%s%s " ,
parent ? parent - > key - > name : " " ,
parent ? " \\ " : " " ,
name ) ;
}
if ( key - > name = = NULL ) {
result = WERR_NOMEM ;
goto done ;
}
/* Tag this as a Performance Counter Key */
2011-05-13 20:23:36 +02:00
if ( strncasecmp_m ( key - > name , KEY_HKPD , strlen ( KEY_HKPD ) ) = = 0 )
2007-06-22 11:03:48 +00:00
key - > type = REG_KEY_HKPD ;
2010-02-07 11:05:07 +01:00
2007-06-22 11:03:48 +00:00
/* Look up the table of registry I/O operations */
2008-04-13 00:54:44 +02:00
if ( ! ( key - > ops = reghook_cache_find ( key - > name ) ) ) {
DEBUG ( 0 , ( " reg_open_onelevel: Failed to assign "
2009-03-23 23:14:45 +01:00
" registry_ops to [%s] \n " , key - > name ) ) ;
2007-06-22 11:03:48 +00:00
result = WERR_BADFILE ;
goto done ;
}
2008-01-15 14:55:04 +01:00
2007-06-22 11:03:48 +00:00
/* check if the path really exists; failed is indicated by -1 */
/* if the subkey count failed, bail out */
2009-02-24 23:17:05 +01:00
result = regsubkey_ctr_init ( key , & subkeys ) ;
if ( ! W_ERROR_IS_OK ( result ) ) {
2007-06-22 11:03:48 +00:00
goto done ;
}
if ( fetch_reg_keys ( key , subkeys ) = = - 1 ) {
result = WERR_BADFILE ;
goto done ;
}
2008-01-15 14:55:04 +01:00
2007-06-22 11:03:48 +00:00
TALLOC_FREE ( subkeys ) ;
if ( ! regkey_access_check ( key , access_desired , & key - > access_granted ,
token ) ) {
result = WERR_ACCESS_DENIED ;
goto done ;
}
* pregkey = regkey ;
result = WERR_OK ;
2010-02-07 11:05:07 +01:00
2007-06-22 11:03:48 +00:00
done :
if ( ! W_ERROR_IS_OK ( result ) ) {
TALLOC_FREE ( regkey ) ;
}
return result ;
}
2006-12-01 20:01:09 +00:00
WERROR reg_openhive ( TALLOC_CTX * mem_ctx , const char * hive ,
uint32 desired_access ,
2010-08-26 20:04:11 +10:00
const struct security_token * token ,
2006-12-01 20:01:09 +00:00
struct registry_key * * pkey )
{
2006-12-05 07:36:14 +00:00
SMB_ASSERT ( hive ! = NULL ) ;
2006-12-01 20:01:09 +00:00
SMB_ASSERT ( hive [ 0 ] ! = ' \0 ' ) ;
SMB_ASSERT ( strchr ( hive , ' \\ ' ) = = NULL ) ;
2006-12-05 07:36:14 +00:00
return regkey_open_onelevel ( mem_ctx , NULL , hive , token , desired_access ,
pkey ) ;
2006-12-01 20:01:09 +00:00
}
2008-01-17 00:57:53 +01:00
/**********************************************************************
* The API functions
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-12-01 20:01:09 +00:00
WERROR reg_openkey ( TALLOC_CTX * mem_ctx , struct registry_key * parent ,
const char * name , uint32 desired_access ,
struct registry_key * * pkey )
{
2006-12-05 07:36:14 +00:00
struct registry_key * direct_parent = parent ;
2006-12-01 20:01:09 +00:00
WERROR err ;
2006-12-05 07:36:14 +00:00
char * p , * path , * to_free ;
size_t len ;
2006-12-01 20:01:09 +00:00
2006-12-05 07:36:14 +00:00
if ( ! ( path = SMB_STRDUP ( name ) ) ) {
2006-12-01 20:01:09 +00:00
return WERR_NOMEM ;
}
2006-12-05 07:36:14 +00:00
to_free = path ;
2006-12-01 20:01:09 +00:00
2006-12-05 07:36:14 +00:00
len = strlen ( path ) ;
2006-12-01 20:01:09 +00:00
2006-12-05 07:36:14 +00:00
if ( ( len > 0 ) & & ( path [ len - 1 ] = = ' \\ ' ) ) {
path [ len - 1 ] = ' \0 ' ;
2006-12-01 20:01:09 +00:00
}
2006-12-05 07:36:14 +00:00
while ( ( p = strchr ( path , ' \\ ' ) ) ! = NULL ) {
char * name_component ;
struct registry_key * tmp ;
2006-12-01 20:01:09 +00:00
2006-12-05 07:36:14 +00:00
if ( ! ( name_component = SMB_STRNDUP ( path , ( p - path ) ) ) ) {
err = WERR_NOMEM ;
goto error ;
}
2006-12-01 20:01:09 +00:00
2006-12-05 07:36:14 +00:00
err = regkey_open_onelevel ( mem_ctx , direct_parent ,
name_component , parent - > token ,
2009-04-15 01:30:12 +02:00
KEY_ENUMERATE_SUB_KEYS , & tmp ) ;
2006-12-05 07:36:14 +00:00
SAFE_FREE ( name_component ) ;
if ( ! W_ERROR_IS_OK ( err ) ) {
goto error ;
}
if ( direct_parent ! = parent ) {
TALLOC_FREE ( direct_parent ) ;
}
direct_parent = tmp ;
path = p + 1 ;
2006-12-01 20:01:09 +00:00
}
2006-12-05 07:36:14 +00:00
err = regkey_open_onelevel ( mem_ctx , direct_parent , path , parent - > token ,
desired_access , pkey ) ;
error :
if ( direct_parent ! = parent ) {
TALLOC_FREE ( direct_parent ) ;
}
SAFE_FREE ( to_free ) ;
return err ;
2006-12-01 20:01:09 +00:00
}
WERROR reg_enumkey ( TALLOC_CTX * mem_ctx , struct registry_key * key ,
uint32 idx , char * * name , NTTIME * last_write_time )
{
WERROR err ;
2009-04-15 01:30:12 +02:00
if ( ! ( key - > key - > access_granted & KEY_ENUMERATE_SUB_KEYS ) ) {
2006-12-01 20:01:09 +00:00
return WERR_ACCESS_DENIED ;
}
if ( ! W_ERROR_IS_OK ( err = fill_subkey_cache ( key ) ) ) {
return err ;
}
2009-02-24 17:30:23 +01:00
if ( idx > = regsubkey_ctr_numkeys ( key - > subkeys ) ) {
2006-12-01 20:01:09 +00:00
return WERR_NO_MORE_ITEMS ;
}
2009-02-24 21:12:26 +01:00
if ( ! ( * name = talloc_strdup ( mem_ctx ,
regsubkey_ctr_specific_key ( key - > subkeys , idx ) ) ) )
{
2006-12-01 20:01:09 +00:00
return WERR_NOMEM ;
}
if ( last_write_time ) {
* last_write_time = 0 ;
}
return WERR_OK ;
}
WERROR reg_enumvalue ( TALLOC_CTX * mem_ctx , struct registry_key * key ,
2006-12-02 10:51:54 +00:00
uint32 idx , char * * pname , struct registry_value * * pval )
2006-12-01 20:01:09 +00:00
{
struct registry_value * val ;
2010-05-23 23:56:40 +02:00
struct regval_blob * blob ;
2006-12-01 20:01:09 +00:00
WERROR err ;
2009-04-15 01:30:12 +02:00
if ( ! ( key - > key - > access_granted & KEY_QUERY_VALUE ) ) {
2006-12-01 20:01:09 +00:00
return WERR_ACCESS_DENIED ;
}
if ( ! ( W_ERROR_IS_OK ( err = fill_value_cache ( key ) ) ) ) {
return err ;
}
2010-05-23 23:56:40 +02:00
if ( idx > = regval_ctr_numvals ( key - > values ) ) {
2007-04-24 00:12:28 +00:00
return WERR_NO_MORE_ITEMS ;
2006-12-01 20:01:09 +00:00
}
2010-05-23 23:56:40 +02:00
blob = regval_ctr_specific_value ( key - > values , idx ) ;
2010-06-29 16:13:15 +02:00
val = talloc_zero ( mem_ctx , struct registry_value ) ;
if ( val = = NULL ) {
return WERR_NOMEM ;
2006-12-01 20:01:09 +00:00
}
2010-06-29 16:13:15 +02:00
val - > type = regval_type ( blob ) ;
val - > data = data_blob_talloc ( mem_ctx , regval_data_p ( blob ) , regval_size ( blob ) ) ;
2006-12-01 20:01:09 +00:00
if ( pname
& & ! ( * pname = talloc_strdup (
2010-05-23 23:56:40 +02:00
mem_ctx , regval_name ( blob ) ) ) ) {
2010-07-01 01:22:11 +02:00
TALLOC_FREE ( val ) ;
2006-12-01 20:01:09 +00:00
return WERR_NOMEM ;
}
2008-01-15 14:55:04 +01:00
2006-12-01 20:01:09 +00:00
* pval = val ;
return WERR_OK ;
}
WERROR reg_queryvalue ( TALLOC_CTX * mem_ctx , struct registry_key * key ,
const char * name , struct registry_value * * pval )
{
WERROR err ;
uint32 i ;
2009-04-15 01:30:12 +02:00
if ( ! ( key - > key - > access_granted & KEY_QUERY_VALUE ) ) {
2006-12-01 20:01:09 +00:00
return WERR_ACCESS_DENIED ;
}
if ( ! ( W_ERROR_IS_OK ( err = fill_value_cache ( key ) ) ) ) {
return err ;
}
2010-05-23 23:56:40 +02:00
for ( i = 0 ; i < regval_ctr_numvals ( key - > values ) ; i + + ) {
struct regval_blob * blob ;
blob = regval_ctr_specific_value ( key - > values , i ) ;
if ( strequal ( regval_name ( blob ) , name ) ) {
2006-12-01 20:01:09 +00:00
return reg_enumvalue ( mem_ctx , key , i , NULL , pval ) ;
}
}
return WERR_BADFILE ;
}
2010-06-30 02:02:43 +02:00
WERROR reg_querymultiplevalues ( TALLOC_CTX * mem_ctx ,
struct registry_key * key ,
uint32_t num_names ,
const char * * names ,
uint32_t * pnum_vals ,
struct registry_value * * pvals )
{
WERROR err ;
uint32_t i , n , found = 0 ;
struct registry_value * vals ;
if ( num_names = = 0 ) {
return WERR_OK ;
}
if ( ! ( key - > key - > access_granted & KEY_QUERY_VALUE ) ) {
return WERR_ACCESS_DENIED ;
}
if ( ! ( W_ERROR_IS_OK ( err = fill_value_cache ( key ) ) ) ) {
return err ;
}
vals = talloc_zero_array ( mem_ctx , struct registry_value , num_names ) ;
if ( vals = = NULL ) {
return WERR_NOMEM ;
}
for ( n = 0 ; n < num_names ; n + + ) {
for ( i = 0 ; i < regval_ctr_numvals ( key - > values ) ; i + + ) {
struct regval_blob * blob ;
blob = regval_ctr_specific_value ( key - > values , i ) ;
if ( strequal ( regval_name ( blob ) , names [ n ] ) ) {
struct registry_value * v ;
err = reg_enumvalue ( mem_ctx , key , i , NULL , & v ) ;
if ( ! W_ERROR_IS_OK ( err ) ) {
return err ;
}
vals [ n ] = * v ;
found + + ;
}
}
}
* pvals = vals ;
* pnum_vals = found ;
return WERR_OK ;
}
2006-12-01 20:01:09 +00:00
WERROR reg_queryinfokey ( struct registry_key * key , uint32_t * num_subkeys ,
uint32_t * max_subkeylen , uint32_t * max_subkeysize ,
uint32_t * num_values , uint32_t * max_valnamelen ,
uint32_t * max_valbufsize , uint32_t * secdescsize ,
NTTIME * last_changed_time )
{
uint32 i , max_size ;
size_t max_len ;
TALLOC_CTX * mem_ctx ;
WERROR err ;
struct security_descriptor * secdesc ;
2009-04-15 01:30:12 +02:00
if ( ! ( key - > key - > access_granted & KEY_QUERY_VALUE ) ) {
2006-12-01 20:01:09 +00:00
return WERR_ACCESS_DENIED ;
}
if ( ! W_ERROR_IS_OK ( fill_subkey_cache ( key ) ) | |
! W_ERROR_IS_OK ( fill_value_cache ( key ) ) ) {
return WERR_BADFILE ;
}
max_len = 0 ;
2009-02-24 17:30:23 +01:00
for ( i = 0 ; i < regsubkey_ctr_numkeys ( key - > subkeys ) ; i + + ) {
2009-02-24 21:12:26 +01:00
max_len = MAX ( max_len ,
strlen ( regsubkey_ctr_specific_key ( key - > subkeys , i ) ) ) ;
2006-12-01 20:01:09 +00:00
}
2009-02-24 17:30:23 +01:00
* num_subkeys = regsubkey_ctr_numkeys ( key - > subkeys ) ;
2006-12-01 20:01:09 +00:00
* max_subkeylen = max_len ;
* max_subkeysize = 0 ; /* Class length? */
max_len = 0 ;
max_size = 0 ;
2010-05-23 23:56:40 +02:00
for ( i = 0 ; i < regval_ctr_numvals ( key - > values ) ; i + + ) {
struct regval_blob * blob ;
blob = regval_ctr_specific_value ( key - > values , i ) ;
max_len = MAX ( max_len , strlen ( regval_name ( blob ) ) ) ;
max_size = MAX ( max_size , regval_size ( blob ) ) ;
2006-12-01 20:01:09 +00:00
}
2010-05-23 23:56:40 +02:00
* num_values = regval_ctr_numvals ( key - > values ) ;
2006-12-01 20:01:09 +00:00
* max_valnamelen = max_len ;
* max_valbufsize = max_size ;
if ( ! ( mem_ctx = talloc_new ( key ) ) ) {
return WERR_NOMEM ;
}
err = regkey_get_secdesc ( mem_ctx , key - > key , & secdesc ) ;
if ( ! W_ERROR_IS_OK ( err ) ) {
TALLOC_FREE ( mem_ctx ) ;
return err ;
}
2010-05-10 00:42:06 +02:00
* secdescsize = ndr_size_security_descriptor ( secdesc , 0 ) ;
2006-12-01 20:01:09 +00:00
TALLOC_FREE ( mem_ctx ) ;
* last_changed_time = 0 ;
return WERR_OK ;
}
WERROR reg_createkey ( TALLOC_CTX * ctx , struct registry_key * parent ,
const char * subkeypath , uint32 desired_access ,
struct registry_key * * pkey ,
enum winreg_CreateAction * paction )
{
struct registry_key * key = parent ;
struct registry_key * create_parent ;
TALLOC_CTX * mem_ctx ;
char * path , * end ;
WERROR err ;
if ( ! ( mem_ctx = talloc_new ( ctx ) ) ) return WERR_NOMEM ;
if ( ! ( path = talloc_strdup ( mem_ctx , subkeypath ) ) ) {
err = WERR_NOMEM ;
goto done ;
}
while ( ( end = strchr ( path , ' \\ ' ) ) ! = NULL ) {
struct registry_key * tmp ;
enum winreg_CreateAction action ;
* end = ' \0 ' ;
err = reg_createkey ( mem_ctx , key , path ,
2009-04-15 01:30:12 +02:00
KEY_ENUMERATE_SUB_KEYS , & tmp , & action ) ;
2006-12-01 20:01:09 +00:00
if ( ! W_ERROR_IS_OK ( err ) ) {
goto done ;
}
if ( key ! = parent ) {
TALLOC_FREE ( key ) ;
}
key = tmp ;
path = end + 1 ;
}
/*
* At this point , " path " contains the one - element subkey of " key " . We
* can try to open it .
*/
err = reg_openkey ( ctx , key , path , desired_access , pkey ) ;
if ( W_ERROR_IS_OK ( err ) ) {
2006-12-03 16:51:31 +00:00
if ( paction ! = NULL ) {
* paction = REG_OPENED_EXISTING_KEY ;
}
goto done ;
2006-12-01 20:01:09 +00:00
}
if ( ! W_ERROR_EQUAL ( err , WERR_BADFILE ) ) {
/*
* Something but " notfound " has happened , so bail out
*/
goto done ;
}
/*
* We have to make a copy of the current key , as we opened it only
* with ENUM_SUBKEY access .
*/
2009-04-15 01:30:12 +02:00
err = reg_openkey ( mem_ctx , key , " " , KEY_CREATE_SUB_KEY ,
2006-12-01 20:01:09 +00:00
& create_parent ) ;
if ( ! W_ERROR_IS_OK ( err ) ) {
goto done ;
}
/*
* Actually create the subkey
*/
err = fill_subkey_cache ( create_parent ) ;
if ( ! W_ERROR_IS_OK ( err ) ) goto done ;
2009-02-25 22:06:47 +01:00
err = create_reg_subkey ( key - > key , path ) ;
W_ERROR_NOT_OK_GOTO_DONE ( err ) ;
2006-12-01 20:01:09 +00:00
/*
* Now open the newly created key
*/
err = reg_openkey ( ctx , create_parent , path , desired_access , pkey ) ;
if ( W_ERROR_IS_OK ( err ) & & ( paction ! = NULL ) ) {
* paction = REG_CREATED_NEW_KEY ;
}
done :
TALLOC_FREE ( mem_ctx ) ;
return err ;
}
2011-08-01 15:27:46 +02:00
static WERROR reg_deletekey_internal ( TALLOC_CTX * mem_ctx ,
struct registry_key * parent ,
const char * path , bool lazy )
2006-12-01 20:01:09 +00:00
{
WERROR err ;
char * name , * end ;
2011-08-01 15:27:46 +02:00
struct registry_key * key ;
2009-02-25 23:13:07 +01:00
name = talloc_strdup ( mem_ctx , path ) ;
if ( name = = NULL ) {
2006-12-01 20:01:09 +00:00
err = WERR_NOMEM ;
2009-02-25 23:13:07 +01:00
goto done ;
2006-12-01 20:01:09 +00:00
}
2007-06-21 22:18:42 +00:00
/* no subkeys - proceed with delete */
2009-02-25 23:13:07 +01:00
end = strrchr ( name , ' \\ ' ) ;
if ( end ! = NULL ) {
2006-12-01 20:01:09 +00:00
* end = ' \0 ' ;
err = reg_openkey ( mem_ctx , parent , name ,
2011-08-01 15:27:46 +02:00
KEY_CREATE_SUB_KEY , & key ) ;
2009-02-25 23:13:07 +01:00
W_ERROR_NOT_OK_GOTO_DONE ( err ) ;
2006-12-01 20:01:09 +00:00
2011-08-01 15:27:46 +02:00
parent = key ;
2006-12-01 20:01:09 +00:00
name = end + 1 ;
}
if ( name [ 0 ] = = ' \0 ' ) {
err = WERR_INVALID_PARAM ;
2009-02-25 23:13:07 +01:00
goto done ;
2006-12-01 20:01:09 +00:00
}
2011-08-01 15:27:46 +02:00
err = delete_reg_subkey ( parent - > key , name , lazy ) ;
done :
return err ;
}
WERROR reg_deletekey ( struct registry_key * parent , const char * path )
{
WERROR err ;
struct registry_key * key ;
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
/* check if the key has subkeys */
err = reg_openkey ( mem_ctx , parent , path , REG_KEY_READ , & key ) ;
W_ERROR_NOT_OK_GOTO_DONE ( err ) ;
err = fill_subkey_cache ( key ) ;
W_ERROR_NOT_OK_GOTO_DONE ( err ) ;
2007-11-06 00:50:47 +01:00
2011-08-01 15:27:46 +02:00
if ( regsubkey_ctr_numkeys ( key - > subkeys ) > 0 ) {
err = WERR_ACCESS_DENIED ;
goto done ;
}
err = reg_deletekey_internal ( mem_ctx , parent , path , false ) ;
2009-02-25 23:13:07 +01:00
done :
2006-12-01 20:01:09 +00:00
TALLOC_FREE ( mem_ctx ) ;
return err ;
}
2011-08-01 15:27:46 +02:00
2006-12-01 20:01:09 +00:00
WERROR reg_setvalue ( struct registry_key * key , const char * name ,
const struct registry_value * val )
{
2011-02-15 18:16:43 +01:00
struct regval_blob * existing ;
2006-12-01 20:01:09 +00:00
WERROR err ;
int res ;
2009-04-15 01:30:12 +02:00
if ( ! ( key - > key - > access_granted & KEY_SET_VALUE ) ) {
2006-12-01 20:01:09 +00:00
return WERR_ACCESS_DENIED ;
}
if ( ! W_ERROR_IS_OK ( err = fill_value_cache ( key ) ) ) {
return err ;
}
2011-02-15 18:16:43 +01:00
existing = regval_ctr_getvalue ( key - > values , name ) ;
if ( ( existing ! = NULL ) & &
( regval_size ( existing ) = = val - > data . length ) & &
( memcmp ( regval_data_p ( existing ) , val - > data . data ,
val - > data . length ) = = 0 ) ) {
return WERR_OK ;
}
2006-12-01 20:01:09 +00:00
res = regval_ctr_addvalue ( key - > values , name , val - > type ,
2010-06-29 16:13:15 +02:00
val - > data . data , val - > data . length ) ;
2006-12-01 20:01:09 +00:00
if ( res = = 0 ) {
TALLOC_FREE ( key - > values ) ;
return WERR_NOMEM ;
}
if ( ! store_reg_values ( key - > key , key - > values ) ) {
TALLOC_FREE ( key - > values ) ;
return WERR_REG_IO_FAILURE ;
}
return WERR_OK ;
}
2008-03-31 17:20:07 +02:00
static WERROR reg_value_exists ( struct registry_key * key , const char * name )
{
2010-05-23 23:56:40 +02:00
struct regval_blob * blob ;
2008-03-31 17:20:07 +02:00
2010-05-23 23:56:40 +02:00
blob = regval_ctr_getvalue ( key - > values , name ) ;
2008-03-31 17:20:07 +02:00
2010-05-23 23:56:40 +02:00
if ( blob = = NULL ) {
return WERR_BADFILE ;
} else {
return WERR_OK ;
}
2008-03-31 17:20:07 +02:00
}
2006-12-01 20:01:09 +00:00
WERROR reg_deletevalue ( struct registry_key * key , const char * name )
{
WERROR err ;
2009-04-15 01:30:12 +02:00
if ( ! ( key - > key - > access_granted & KEY_SET_VALUE ) ) {
2006-12-01 20:01:09 +00:00
return WERR_ACCESS_DENIED ;
}
if ( ! W_ERROR_IS_OK ( err = fill_value_cache ( key ) ) ) {
return err ;
}
2008-03-31 17:20:07 +02:00
err = reg_value_exists ( key , name ) ;
if ( ! W_ERROR_IS_OK ( err ) ) {
return err ;
}
2006-12-01 20:01:09 +00:00
regval_ctr_delvalue ( key - > values , name ) ;
if ( ! store_reg_values ( key - > key , key - > values ) ) {
TALLOC_FREE ( key - > values ) ;
return WERR_REG_IO_FAILURE ;
}
return WERR_OK ;
}
2008-01-17 11:02:15 +01:00
WERROR reg_getkeysecurity ( TALLOC_CTX * mem_ctx , struct registry_key * key ,
struct security_descriptor * * psecdesc )
{
return regkey_get_secdesc ( mem_ctx , key - > key , psecdesc ) ;
}
WERROR reg_setkeysecurity ( struct registry_key * key ,
struct security_descriptor * psecdesc )
{
return regkey_set_secdesc ( key - > key , psecdesc ) ;
}
2008-01-17 00:57:53 +01:00
2008-01-17 11:22:01 +01:00
WERROR reg_getversion ( uint32_t * version )
{
if ( version = = NULL ) {
return WERR_INVALID_PARAM ;
}
* version = 0x00000005 ; /* Windows 2000 registry API version */
return WERR_OK ;
}
2008-01-17 00:57:53 +01:00
/**********************************************************************
* Higher level utility functions
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-08-14 15:29:42 +00:00
WERROR reg_deleteallvalues ( struct registry_key * key )
{
WERROR err ;
int i ;
2009-04-15 01:30:12 +02:00
if ( ! ( key - > key - > access_granted & KEY_SET_VALUE ) ) {
2007-08-14 15:29:42 +00:00
return WERR_ACCESS_DENIED ;
}
if ( ! W_ERROR_IS_OK ( err = fill_value_cache ( key ) ) ) {
return err ;
}
2010-05-23 23:56:40 +02:00
for ( i = 0 ; i < regval_ctr_numvals ( key - > values ) ; i + + ) {
struct regval_blob * blob ;
blob = regval_ctr_specific_value ( key - > values , i ) ;
regval_ctr_delvalue ( key - > values , regval_name ( blob ) ) ;
2007-08-14 15:29:42 +00:00
}
if ( ! store_reg_values ( key - > key , key - > values ) ) {
TALLOC_FREE ( key - > values ) ;
return WERR_REG_IO_FAILURE ;
}
return WERR_OK ;
}
2007-04-09 10:38:55 +00:00
2007-06-22 11:21:59 +00:00
/*
2008-01-15 14:55:04 +01:00
* Utility function to delete a registry key with all its subkeys .
* Note that reg_deletekey returns ACCESS_DENIED when called on a
2007-06-22 11:21:59 +00:00
* key that has subkeys .
*/
2010-09-22 06:21:38 +02:00
static WERROR reg_deletekey_recursive_internal ( struct registry_key * parent ,
2008-01-17 00:16:58 +01:00
const char * path ,
2011-08-01 15:27:46 +02:00
bool del_key , bool lazy )
2007-06-22 11:21:59 +00:00
{
WERROR werr = WERR_OK ;
struct registry_key * key ;
char * subkey_name = NULL ;
2009-02-26 02:59:07 +01:00
uint32 i ;
2010-09-22 06:21:38 +02:00
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
2007-06-22 11:21:59 +00:00
2011-08-14 23:48:41 +02:00
DEBUG ( 5 , ( " reg_deletekey_recursive_internal: deleting '%s' from '%s' \n " ,
path , parent - > key - > name ) ) ;
2007-06-22 11:21:59 +00:00
/* recurse through subkeys first */
2007-12-31 03:25:54 +01:00
werr = reg_openkey ( mem_ctx , parent , path , REG_KEY_ALL , & key ) ;
2007-06-22 11:21:59 +00:00
if ( ! W_ERROR_IS_OK ( werr ) ) {
2011-08-14 23:48:41 +02:00
DEBUG ( 3 , ( " reg_deletekey_recursive_internal: error opening "
" subkey '%s' of '%s': '%s' \n " ,
path , parent - > key - > name , win_errstr ( werr ) ) ) ;
2007-06-22 11:21:59 +00:00
goto done ;
}
2009-02-26 02:59:07 +01:00
werr = fill_subkey_cache ( key ) ;
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
/*
* loop from top to bottom for perfomance :
* this way , we need to rehash the regsubkey containers less
*/
for ( i = regsubkey_ctr_numkeys ( key - > subkeys ) ; i > 0 ; i - - ) {
subkey_name = regsubkey_ctr_specific_key ( key - > subkeys , i - 1 ) ;
2011-08-01 15:27:46 +02:00
werr = reg_deletekey_recursive_internal ( key , subkey_name , true , del_key ) ;
2009-02-26 02:59:07 +01:00
W_ERROR_NOT_OK_GOTO_DONE ( werr ) ;
2007-06-22 11:21:59 +00:00
}
2007-09-27 01:26:19 +00:00
if ( del_key ) {
/* now delete the actual key */
2011-08-01 15:27:46 +02:00
werr = reg_deletekey_internal ( mem_ctx , parent , path , lazy ) ;
2007-09-27 01:26:19 +00:00
}
2007-06-22 11:21:59 +00:00
done :
2011-08-14 23:48:41 +02:00
DEBUG ( 5 , ( " reg_deletekey_recursive_internal: done deleting '%s' from "
" '%s': %s \n " ,
path , parent - > key - > name , win_errstr ( werr ) ) ) ;
2007-06-22 11:21:59 +00:00
TALLOC_FREE ( mem_ctx ) ;
return werr ;
}
2007-09-27 01:26:19 +00:00
2010-09-22 06:21:38 +02:00
static WERROR reg_deletekey_recursive_trans ( struct registry_key * parent ,
2009-02-24 15:40:05 +01:00
const char * path ,
bool del_key )
{
WERROR werr ;
werr = regdb_transaction_start ( ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
DEBUG ( 0 , ( " reg_deletekey_recursive_trans: "
" error starting transaction: %s \n " ,
win_errstr ( werr ) ) ) ;
return werr ;
}
2011-08-01 15:27:46 +02:00
werr = reg_deletekey_recursive_internal ( parent , path , del_key , false ) ;
2009-02-24 15:40:05 +01:00
if ( ! W_ERROR_IS_OK ( werr ) ) {
2011-01-30 13:44:31 +01:00
WERROR werr2 ;
2009-02-25 09:53:16 +01:00
DEBUG ( 1 , ( __location__ " failed to delete key '%s' from key "
" '%s': %s \n " , path , parent - > key - > name ,
win_errstr ( werr ) ) ) ;
2011-01-30 13:44:31 +01:00
werr2 = regdb_transaction_cancel ( ) ;
if ( ! W_ERROR_IS_OK ( werr2 ) ) {
2009-02-24 15:40:05 +01:00
DEBUG ( 0 , ( " reg_deletekey_recursive_trans: "
" error cancelling transaction: %s \n " ,
2011-01-30 13:44:31 +01:00
win_errstr ( werr2 ) ) ) ;
/*
* return the original werr or the
* error from cancelling the transaction ?
*/
2009-02-24 15:40:05 +01:00
}
} else {
werr = regdb_transaction_commit ( ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
DEBUG ( 0 , ( " reg_deletekey_recursive_trans: "
" error committing transaction: %s \n " ,
win_errstr ( werr ) ) ) ;
2011-08-14 23:48:41 +02:00
} else {
DEBUG ( 5 , ( " reg_reletekey_recursive_trans: deleted key '%s' from '%s' \n " ,
path , parent - > key - > name ) ) ;
2009-02-24 15:40:05 +01:00
}
}
return werr ;
}
2010-09-22 06:21:38 +02:00
WERROR reg_deletekey_recursive ( struct registry_key * parent ,
2007-09-27 01:26:19 +00:00
const char * path )
{
2010-09-22 06:21:38 +02:00
return reg_deletekey_recursive_trans ( parent , path , true ) ;
2007-09-27 01:26:19 +00:00
}
2010-09-22 06:21:38 +02:00
WERROR reg_deletesubkeys_recursive ( struct registry_key * parent ,
2007-09-27 01:26:19 +00:00
const char * path )
{
2010-09-22 06:21:38 +02:00
return reg_deletekey_recursive_trans ( parent , path , false ) ;
2007-09-27 01:26:19 +00:00
}
2008-01-17 10:19:12 +01:00