2007-10-06 00:17:44 +00:00
/*
2005-09-29 12:00:49 +00:00
Unix SMB / CIFS implementation .
2007-08-26 15:16:40 +00:00
Reading registry patch files
2007-10-06 00:17:44 +00:00
2007-08-26 15:16:40 +00:00
Copyright ( C ) Jelmer Vernooij 2004 - 2007
Copyright ( C ) Wilco Baan Hofman 2006
2010-03-22 12:42:21 +01:00
Copyright ( C ) Matthias Dieter Wallnöfer 2008 - 2010
2005-09-29 12:00:49 +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-10 02:07:03 +00:00
the Free Software Foundation ; either version 3 of the License , or
2005-09-29 12:00:49 +00:00
( at your option ) any later version .
2007-10-06 00:17:44 +00:00
2005-09-29 12:00:49 +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 .
2007-10-06 00:17:44 +00:00
2005-09-29 12:00:49 +00:00
You should have received a copy of the GNU General Public License
2007-07-10 02:07:03 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2005-09-29 12:00:49 +00:00
*/
# include "includes.h"
# include "lib/registry/registry.h"
# include "system/filesys.h"
2007-10-06 00:17:44 +00:00
_PUBLIC_ WERROR reg_preg_diff_load ( int fd ,
const struct reg_diff_callbacks * callbacks ,
void * callback_data ) ;
2005-09-29 12:00:49 +00:00
2007-10-06 00:17:44 +00:00
_PUBLIC_ WERROR reg_dotreg_diff_load ( int fd ,
const struct reg_diff_callbacks * callbacks ,
void * callback_data ) ;
2005-09-29 12:00:49 +00:00
/*
* Generate difference between two keys
*/
2007-10-06 00:17:44 +00:00
WERROR reg_generate_diff_key ( struct registry_key * oldkey ,
struct registry_key * newkey ,
const char * path ,
const struct reg_diff_callbacks * callbacks ,
void * callback_data )
2005-09-29 12:00:49 +00:00
{
2009-11-07 21:07:20 +01:00
unsigned int i ;
2008-04-15 11:52:33 +02:00
struct registry_key * t1 = NULL , * t2 = NULL ;
2007-08-26 15:16:40 +00:00
char * tmppath ;
const char * keyname1 ;
WERROR error , error1 , error2 ;
2005-09-29 12:00:49 +00:00
TALLOC_CTX * mem_ctx = talloc_init ( " writediff " ) ;
2007-08-26 15:16:40 +00:00
uint32_t old_num_subkeys , old_num_values ,
new_num_subkeys , new_num_values ;
2005-09-29 12:00:49 +00:00
2007-08-26 15:16:40 +00:00
if ( oldkey ! = NULL ) {
2007-10-06 00:17:44 +00:00
error = reg_key_get_info ( mem_ctx , oldkey , NULL ,
& old_num_subkeys , & old_num_values ,
2008-01-07 14:11:29 -06:00
NULL , NULL , NULL , NULL ) ;
2007-08-26 15:16:40 +00:00
if ( ! W_ERROR_IS_OK ( error ) ) {
2008-09-15 19:00:07 +02:00
DEBUG ( 0 , ( " Error occurred while getting key info: %s \n " ,
2007-10-06 00:17:44 +00:00
win_errstr ( error ) ) ) ;
2008-09-15 19:00:07 +02:00
talloc_free ( mem_ctx ) ;
2007-08-26 15:16:40 +00:00
return error ;
}
} else {
old_num_subkeys = 0 ;
old_num_values = 0 ;
}
2005-09-29 12:00:49 +00:00
2008-10-03 14:29:47 +02:00
/* Subkeys that were changed or deleted */
2007-08-26 15:16:40 +00:00
for ( i = 0 ; i < old_num_subkeys ; i + + ) {
2007-10-06 00:17:44 +00:00
error1 = reg_key_get_subkey_by_index ( mem_ctx , oldkey , i ,
2008-09-15 19:00:07 +02:00
& keyname1 , NULL , NULL ) ;
2007-08-26 15:16:40 +00:00
if ( ! W_ERROR_IS_OK ( error1 ) ) {
2008-09-15 19:00:07 +02:00
DEBUG ( 0 , ( " Error occurred while getting subkey by index: %s \n " ,
win_errstr ( error1 ) ) ) ;
2005-09-29 12:00:49 +00:00
continue ;
2007-08-26 15:16:40 +00:00
}
if ( newkey ! = NULL ) {
error2 = reg_open_key ( mem_ctx , newkey , keyname1 , & t2 ) ;
} else {
2015-12-03 15:24:12 +01:00
error2 = WERR_FILE_NOT_FOUND ;
2007-08-26 15:16:40 +00:00
t2 = NULL ;
}
2005-09-29 12:00:49 +00:00
2015-12-03 15:24:12 +01:00
if ( ! W_ERROR_IS_OK ( error2 ) & & ! W_ERROR_EQUAL ( error2 , WERR_FILE_NOT_FOUND ) ) {
2010-02-21 17:24:08 +11:00
DEBUG ( 0 , ( " Error occurred while getting subkey by name: %s \n " ,
2007-10-06 00:17:44 +00:00
win_errstr ( error2 ) ) ) ;
2007-08-26 15:16:40 +00:00
talloc_free ( mem_ctx ) ;
2005-09-29 12:00:49 +00:00
return error2 ;
}
2015-12-03 15:24:12 +01:00
/* if "error2" is going to be "WERR_FILE_NOT_FOUND", then newkey */
2008-10-03 14:29:47 +02:00
/* didn't have such a subkey and therefore add a del diff */
2007-08-26 15:16:40 +00:00
tmppath = talloc_asprintf ( mem_ctx , " %s \\ %s " , path , keyname1 ) ;
2010-03-22 13:39:40 +01:00
if ( tmppath = = NULL ) {
DEBUG ( 0 , ( " Out of memory \n " ) ) ;
talloc_free ( mem_ctx ) ;
2015-12-03 15:24:19 +01:00
return WERR_NOT_ENOUGH_MEMORY ;
2010-03-22 13:39:40 +01:00
}
2008-10-03 14:29:47 +02:00
if ( ! W_ERROR_IS_OK ( error2 ) )
callbacks - > del_key ( callback_data , tmppath ) ;
2008-09-15 19:00:07 +02:00
/* perform here also the recursive invocation */
error1 = reg_open_key ( mem_ctx , oldkey , keyname1 , & t1 ) ;
if ( ! W_ERROR_IS_OK ( error1 ) ) {
2010-02-21 17:24:08 +11:00
DEBUG ( 0 , ( " Error occurred while getting subkey by name: %s \n " ,
2008-09-15 19:00:07 +02:00
win_errstr ( error1 ) ) ) ;
talloc_free ( mem_ctx ) ;
return error1 ;
}
reg_generate_diff_key ( t1 , t2 , tmppath , callbacks , callback_data ) ;
2007-10-06 00:17:44 +00:00
talloc_free ( tmppath ) ;
2005-09-29 12:00:49 +00:00
}
2007-08-26 15:16:40 +00:00
if ( newkey ! = NULL ) {
2007-10-06 00:17:44 +00:00
error = reg_key_get_info ( mem_ctx , newkey , NULL ,
& new_num_subkeys , & new_num_values ,
2008-01-07 14:11:29 -06:00
NULL , NULL , NULL , NULL ) ;
2007-08-26 15:16:40 +00:00
if ( ! W_ERROR_IS_OK ( error ) ) {
2008-09-15 19:00:07 +02:00
DEBUG ( 0 , ( " Error occurred while getting key info: %s \n " ,
2007-10-06 00:17:44 +00:00
win_errstr ( error ) ) ) ;
2008-09-15 19:00:07 +02:00
talloc_free ( mem_ctx ) ;
2007-08-26 15:16:40 +00:00
return error ;
}
} else {
new_num_subkeys = 0 ;
new_num_values = 0 ;
2005-09-29 12:00:49 +00:00
}
/* Subkeys that were added */
2007-08-26 15:16:40 +00:00
for ( i = 0 ; i < new_num_subkeys ; i + + ) {
2008-09-15 19:00:07 +02:00
error1 = reg_key_get_subkey_by_index ( mem_ctx , newkey , i ,
& keyname1 , NULL , NULL ) ;
2007-08-26 15:16:40 +00:00
if ( ! W_ERROR_IS_OK ( error1 ) ) {
2008-09-15 19:00:07 +02:00
DEBUG ( 0 , ( " Error occurred while getting subkey by index: %s \n " ,
2007-10-06 00:17:44 +00:00
win_errstr ( error1 ) ) ) ;
2007-08-26 15:16:40 +00:00
talloc_free ( mem_ctx ) ;
return error1 ;
}
if ( oldkey ! = NULL ) {
error2 = reg_open_key ( mem_ctx , oldkey , keyname1 , & t1 ) ;
2007-10-06 00:17:44 +00:00
2007-08-26 15:16:40 +00:00
if ( W_ERROR_IS_OK ( error2 ) )
continue ;
} else {
2015-12-03 15:24:12 +01:00
error2 = WERR_FILE_NOT_FOUND ;
2007-08-26 15:16:40 +00:00
t1 = NULL ;
}
2007-10-06 00:17:44 +00:00
2015-12-03 15:24:12 +01:00
if ( ! W_ERROR_EQUAL ( error2 , WERR_FILE_NOT_FOUND ) ) {
2008-09-15 19:00:07 +02:00
DEBUG ( 0 , ( " Error occurred while getting subkey by name: %s \n " ,
2007-10-06 00:17:44 +00:00
win_errstr ( error2 ) ) ) ;
2007-08-26 15:16:40 +00:00
talloc_free ( mem_ctx ) ;
2005-09-29 12:00:49 +00:00
return error2 ;
}
2010-03-22 00:02:06 +01:00
/* oldkey didn't have such a subkey, add a add diff */
2007-08-26 15:16:40 +00:00
tmppath = talloc_asprintf ( mem_ctx , " %s \\ %s " , path , keyname1 ) ;
2010-03-22 13:39:40 +01:00
if ( tmppath = = NULL ) {
DEBUG ( 0 , ( " Out of memory \n " ) ) ;
talloc_free ( mem_ctx ) ;
2015-12-03 15:24:19 +01:00
return WERR_NOT_ENOUGH_MEMORY ;
2010-03-22 13:39:40 +01:00
}
2007-08-26 15:16:40 +00:00
callbacks - > add_key ( callback_data , tmppath ) ;
2005-09-29 12:00:49 +00:00
2008-09-15 19:00:07 +02:00
/* perform here also the recursive invocation */
error1 = reg_open_key ( mem_ctx , newkey , keyname1 , & t2 ) ;
if ( ! W_ERROR_IS_OK ( error1 ) ) {
2010-02-21 17:24:08 +11:00
DEBUG ( 0 , ( " Error occurred while getting subkey by name: %s \n " ,
2008-09-15 19:00:07 +02:00
win_errstr ( error1 ) ) ) ;
talloc_free ( mem_ctx ) ;
return error1 ;
}
reg_generate_diff_key ( t1 , t2 , tmppath , callbacks , callback_data ) ;
2007-08-26 15:16:40 +00:00
talloc_free ( tmppath ) ;
2005-09-29 12:00:49 +00:00
}
2008-09-15 19:00:07 +02:00
/* Values that were added or changed */
2007-08-26 15:16:40 +00:00
for ( i = 0 ; i < new_num_values ; i + + ) {
const char * name ;
uint32_t type1 , type2 ;
2010-03-22 13:33:04 +01:00
DATA_BLOB contents1 = { NULL , 0 } , contents2 = { NULL , 0 } ;
2007-08-26 15:16:40 +00:00
2007-10-06 00:17:44 +00:00
error1 = reg_key_get_value_by_index ( mem_ctx , newkey , i ,
& name , & type1 , & contents1 ) ;
2007-08-26 15:16:40 +00:00
if ( ! W_ERROR_IS_OK ( error1 ) ) {
2008-09-15 19:00:07 +02:00
DEBUG ( 0 , ( " Unable to get value by index: %s \n " ,
2007-10-06 00:17:44 +00:00
win_errstr ( error1 ) ) ) ;
2007-08-26 15:16:40 +00:00
talloc_free ( mem_ctx ) ;
return error1 ;
}
if ( oldkey ! = NULL ) {
2007-10-06 00:17:44 +00:00
error2 = reg_key_get_value_by_name ( mem_ctx , oldkey ,
name , & type2 ,
& contents2 ) ;
} else
2015-12-03 15:24:12 +01:00
error2 = WERR_FILE_NOT_FOUND ;
2007-10-06 00:17:44 +00:00
2008-09-15 19:00:07 +02:00
if ( ! W_ERROR_IS_OK ( error2 )
2015-12-03 15:24:12 +01:00
& & ! W_ERROR_EQUAL ( error2 , WERR_FILE_NOT_FOUND ) ) {
2008-09-15 19:00:07 +02:00
DEBUG ( 0 , ( " Error occurred while getting value by name: %s \n " ,
2007-10-06 00:17:44 +00:00
win_errstr ( error2 ) ) ) ;
2007-08-26 15:16:40 +00:00
talloc_free ( mem_ctx ) ;
2005-09-29 12:00:49 +00:00
return error2 ;
}
2008-09-15 19:00:07 +02:00
if ( W_ERROR_IS_OK ( error2 )
2010-03-22 12:42:21 +01:00
& & ( data_blob_cmp ( & contents1 , & contents2 ) = = 0 )
& & ( type1 = = type2 ) ) {
talloc_free ( discard_const_p ( char , name ) ) ;
talloc_free ( contents1 . data ) ;
talloc_free ( contents2 . data ) ;
2005-09-29 12:00:49 +00:00
continue ;
2010-03-22 12:42:21 +01:00
}
2005-09-29 12:00:49 +00:00
2007-10-06 00:17:44 +00:00
callbacks - > set_value ( callback_data , path , name ,
type1 , contents1 ) ;
2010-03-22 12:42:21 +01:00
talloc_free ( discard_const_p ( char , name ) ) ;
talloc_free ( contents1 . data ) ;
talloc_free ( contents2 . data ) ;
2005-09-29 12:00:49 +00:00
}
/* Values that were deleted */
2007-08-26 15:16:40 +00:00
for ( i = 0 ; i < old_num_values ; i + + ) {
const char * name ;
2008-09-15 19:00:07 +02:00
uint32_t type ;
2010-03-22 13:33:04 +01:00
DATA_BLOB contents = { NULL , 0 } ;
2008-09-15 19:00:07 +02:00
2007-10-06 00:17:44 +00:00
error1 = reg_key_get_value_by_index ( mem_ctx , oldkey , i , & name ,
2008-09-15 19:00:07 +02:00
& type , & contents ) ;
2007-08-26 15:16:40 +00:00
if ( ! W_ERROR_IS_OK ( error1 ) ) {
2008-09-15 19:00:07 +02:00
DEBUG ( 0 , ( " Unable to get value by index: %s \n " ,
2007-10-06 00:17:44 +00:00
win_errstr ( error1 ) ) ) ;
2007-08-26 15:16:40 +00:00
talloc_free ( mem_ctx ) ;
return error1 ;
}
2008-09-15 19:00:07 +02:00
if ( newkey ! = NULL )
error2 = reg_key_get_value_by_name ( mem_ctx , newkey ,
name , & type , & contents ) ;
else
2015-12-03 15:24:12 +01:00
error2 = WERR_FILE_NOT_FOUND ;
2005-09-29 12:00:49 +00:00
2010-03-22 12:42:21 +01:00
if ( W_ERROR_IS_OK ( error2 ) ) {
talloc_free ( discard_const_p ( char , name ) ) ;
talloc_free ( contents . data ) ;
2005-09-29 12:00:49 +00:00
continue ;
2010-03-22 12:42:21 +01:00
}
2005-09-29 12:00:49 +00:00
2015-12-03 15:24:12 +01:00
if ( ! W_ERROR_EQUAL ( error2 , WERR_FILE_NOT_FOUND ) ) {
2008-09-15 19:00:07 +02:00
DEBUG ( 0 , ( " Error occurred while getting value by name: %s \n " ,
2007-10-06 00:17:44 +00:00
win_errstr ( error2 ) ) ) ;
2008-09-15 19:00:07 +02:00
talloc_free ( mem_ctx ) ;
2005-09-29 12:00:49 +00:00
return error2 ;
}
2007-08-26 15:16:40 +00:00
callbacks - > del_value ( callback_data , path , name ) ;
2010-03-22 12:42:21 +01:00
talloc_free ( discard_const_p ( char , name ) ) ;
talloc_free ( contents . data ) ;
2005-09-29 12:00:49 +00:00
}
talloc_free ( mem_ctx ) ;
return WERR_OK ;
}
2006-03-06 10:09:53 +00:00
/**
2007-10-06 00:17:44 +00:00
* Generate diff between two registry contexts
2005-09-29 12:00:49 +00:00
*/
2007-10-06 00:17:44 +00:00
_PUBLIC_ WERROR reg_generate_diff ( struct registry_context * ctx1 ,
struct registry_context * ctx2 ,
2007-08-26 15:16:40 +00:00
const struct reg_diff_callbacks * callbacks ,
void * callback_data )
2005-09-29 12:00:49 +00:00
{
2009-11-07 21:07:20 +01:00
unsigned int i ;
2005-09-29 12:00:49 +00:00
WERROR error ;
2008-09-15 19:00:07 +02:00
for ( i = 0 ; reg_predefined_keys [ i ] . name ; i + + ) {
2007-08-26 15:16:40 +00:00
struct registry_key * r1 = NULL , * r2 = NULL ;
2008-09-15 19:00:07 +02:00
error = reg_get_predefined_key ( ctx1 ,
reg_predefined_keys [ i ] . handle , & r1 ) ;
2007-10-06 00:17:44 +00:00
if ( ! W_ERROR_IS_OK ( error ) & &
2015-12-03 15:24:12 +01:00
! W_ERROR_EQUAL ( error , WERR_FILE_NOT_FOUND ) ) {
2007-10-06 00:17:44 +00:00
DEBUG ( 0 , ( " Unable to open hive %s for backend 1 \n " ,
2008-09-15 19:00:07 +02:00
reg_predefined_keys [ i ] . name ) ) ;
continue ;
2005-09-29 12:00:49 +00:00
}
2007-10-06 00:17:44 +00:00
2008-09-15 19:00:07 +02:00
error = reg_get_predefined_key ( ctx2 ,
reg_predefined_keys [ i ] . handle , & r2 ) ;
2007-10-06 00:17:44 +00:00
if ( ! W_ERROR_IS_OK ( error ) & &
2015-12-03 15:24:12 +01:00
! W_ERROR_EQUAL ( error , WERR_FILE_NOT_FOUND ) ) {
2007-10-06 00:17:44 +00:00
DEBUG ( 0 , ( " Unable to open hive %s for backend 2 \n " ,
2008-09-15 19:00:07 +02:00
reg_predefined_keys [ i ] . name ) ) ;
2005-09-29 12:00:49 +00:00
continue ;
2008-09-15 19:00:07 +02:00
}
2005-09-29 12:00:49 +00:00
2010-03-22 11:45:20 +01:00
/* if "r1" is NULL (old hive) and "r2" isn't (new hive) then
* the hive doesn ' t exist yet and we have to generate an add
* diff */
2010-03-22 00:13:02 +01:00
if ( ( r1 = = NULL ) & & ( r2 ! = NULL ) ) {
callbacks - > add_key ( callback_data ,
reg_predefined_keys [ i ] . name ) ;
}
2010-03-22 11:45:20 +01:00
/* if "r1" isn't NULL (old hive) and "r2" is (new hive) then
* the hive shouldn ' t exist anymore and we have to generate a
* del diff */
2010-03-22 00:13:02 +01:00
if ( ( r1 ! = NULL ) & & ( r2 = = NULL ) ) {
callbacks - > del_key ( callback_data ,
reg_predefined_keys [ i ] . name ) ;
}
2008-09-15 19:00:07 +02:00
error = reg_generate_diff_key ( r1 , r2 ,
reg_predefined_keys [ i ] . name , callbacks ,
callback_data ) ;
2007-08-26 15:16:40 +00:00
if ( ! W_ERROR_IS_OK ( error ) ) {
2007-10-06 00:17:44 +00:00
DEBUG ( 0 , ( " Unable to determine diff: %s \n " ,
win_errstr ( error ) ) ) ;
2007-08-26 15:16:40 +00:00
return error ;
2005-09-29 12:00:49 +00:00
}
}
2007-08-26 15:16:40 +00:00
if ( callbacks - > done ! = NULL ) {
callbacks - > done ( callback_data ) ;
}
2005-09-29 12:00:49 +00:00
return WERR_OK ;
}
2006-03-06 10:09:53 +00:00
/**
2005-09-29 12:00:49 +00:00
* Load diff file
*/
2007-10-06 00:17:44 +00:00
_PUBLIC_ WERROR reg_diff_load ( const char * filename ,
const struct reg_diff_callbacks * callbacks ,
2007-09-07 13:31:15 +00:00
void * callback_data )
2005-09-29 12:00:49 +00:00
{
int fd ;
2007-08-26 15:16:40 +00:00
char hdr [ 4 ] ;
2007-10-06 00:17:44 +00:00
2007-08-26 15:16:40 +00:00
fd = open ( filename , O_RDONLY , 0 ) ;
2005-09-29 12:00:49 +00:00
if ( fd = = - 1 ) {
2007-10-06 00:17:44 +00:00
DEBUG ( 0 , ( " Error opening registry patch file `%s' \n " ,
filename ) ) ;
2015-12-03 15:24:21 +01:00
return WERR_GEN_FAILURE ;
2005-09-29 12:00:49 +00:00
}
2007-08-26 15:16:40 +00:00
if ( read ( fd , & hdr , 4 ) ! = 4 ) {
2007-10-06 00:17:44 +00:00
DEBUG ( 0 , ( " Error reading registry patch file `%s' \n " ,
filename ) ) ;
2009-07-18 23:30:33 +07:00
close ( fd ) ;
2015-12-03 15:24:21 +01:00
return WERR_GEN_FAILURE ;
2005-09-29 12:00:49 +00:00
}
2007-08-26 15:16:40 +00:00
/* Reset position in file */
lseek ( fd , 0 , SEEK_SET ) ;
2008-04-15 11:52:33 +02:00
#if 0 /* These backends are not supported yet. */
2007-08-26 15:16:40 +00:00
if ( strncmp ( hdr , " CREG " , 4 ) = = 0 ) {
/* Must be a W9x CREG Config.pol file */
return reg_creg_diff_load ( diff , fd ) ;
} else if ( strncmp ( hdr , " regf " , 4 ) = = 0 ) {
/* Must be a REGF NTConfig.pol file */
return reg_regf_diff_load ( diff , fd ) ;
2007-10-06 00:17:44 +00:00
} else
# endif
2007-08-26 15:16:40 +00:00
if ( strncmp ( hdr , " PReg " , 4 ) = = 0 ) {
/* Must be a GPO Registry.pol file */
2010-05-09 17:20:01 +02:00
return reg_preg_diff_load ( fd , callbacks , callback_data ) ;
2007-08-26 15:16:40 +00:00
} else {
/* Must be a normal .REG file */
2010-05-09 17:20:01 +02:00
return reg_dotreg_diff_load ( fd , callbacks , callback_data ) ;
2005-09-29 12:00:49 +00:00
}
2007-08-26 15:16:40 +00:00
}
2005-09-29 12:00:49 +00:00
2007-08-26 15:16:40 +00:00
/**
* The reg_diff_apply functions
*/
2007-10-06 00:17:44 +00:00
static WERROR reg_diff_apply_add_key ( void * _ctx , const char * key_name )
2007-08-26 15:16:40 +00:00
{
2007-09-07 13:31:15 +00:00
struct registry_context * ctx = ( struct registry_context * ) _ctx ;
2007-08-26 15:16:40 +00:00
struct registry_key * tmp ;
2008-04-15 19:57:29 +02:00
char * buf , * buf_ptr ;
2007-08-26 15:16:40 +00:00
WERROR error ;
2005-09-29 12:00:49 +00:00
2008-04-15 19:57:29 +02:00
/* Recursively create the path */
buf = talloc_strdup ( ctx , key_name ) ;
2010-03-22 22:01:31 +01:00
W_ERROR_HAVE_NO_MEMORY ( buf ) ;
2008-04-15 19:57:29 +02:00
buf_ptr = buf ;
while ( * buf_ptr + + ! = ' \0 ' ) {
if ( * buf_ptr = = ' \\ ' ) {
* buf_ptr = ' \0 ' ;
error = reg_key_add_abs ( ctx , ctx , buf , 0 , NULL , & tmp ) ;
if ( ! W_ERROR_EQUAL ( error , WERR_ALREADY_EXISTS ) & &
! W_ERROR_IS_OK ( error ) ) {
DEBUG ( 0 , ( " Error adding new key '%s': %s \n " ,
key_name , win_errstr ( error ) ) ) ;
return error ;
}
* buf_ptr + + = ' \\ ' ;
2010-03-22 12:42:21 +01:00
talloc_free ( tmp ) ;
2008-04-15 19:57:29 +02:00
}
}
2010-03-22 12:42:21 +01:00
talloc_free ( buf ) ;
2008-04-15 19:57:29 +02:00
/* Add the key */
2007-08-26 15:16:40 +00:00
error = reg_key_add_abs ( ctx , ctx , key_name , 0 , NULL , & tmp ) ;
2005-09-29 12:00:49 +00:00
2007-10-06 00:17:44 +00:00
if ( ! W_ERROR_EQUAL ( error , WERR_ALREADY_EXISTS ) & &
2008-04-15 19:57:29 +02:00
! W_ERROR_IS_OK ( error ) ) {
2007-10-06 00:17:44 +00:00
DEBUG ( 0 , ( " Error adding new key '%s': %s \n " ,
key_name , win_errstr ( error ) ) ) ;
2007-08-26 15:16:40 +00:00
return error ;
}
2010-03-22 12:42:21 +01:00
talloc_free ( tmp ) ;
2007-08-26 15:16:40 +00:00
return WERR_OK ;
}
2005-09-29 12:00:49 +00:00
2007-10-06 00:17:44 +00:00
static WERROR reg_diff_apply_del_key ( void * _ctx , const char * key_name )
2007-08-26 15:16:40 +00:00
{
2007-09-07 13:31:15 +00:00
struct registry_context * ctx = ( struct registry_context * ) _ctx ;
2005-09-29 12:00:49 +00:00
2008-10-03 14:29:47 +02:00
/* We can't proof here for success, because a common superkey could */
/* have been deleted before the subkey's (diff order). This removed */
2015-12-03 15:24:12 +01:00
/* therefore all children recursively and the "WERR_FILE_NOT_FOUND" result is */
2008-10-03 14:29:47 +02:00
/* expected. */
2005-09-29 12:00:49 +00:00
2010-03-29 18:43:38 +02:00
reg_key_del_abs ( ctx , key_name ) ;
2007-10-06 00:17:44 +00:00
2007-08-26 15:16:40 +00:00
return WERR_OK ;
}
2005-09-29 12:00:49 +00:00
2007-10-06 00:17:44 +00:00
static WERROR reg_diff_apply_set_value ( void * _ctx , const char * path ,
const char * value_name ,
uint32_t value_type , DATA_BLOB value )
2007-08-26 15:16:40 +00:00
{
2007-09-07 13:31:15 +00:00
struct registry_context * ctx = ( struct registry_context * ) _ctx ;
2007-08-26 15:16:40 +00:00
struct registry_key * tmp ;
WERROR error ;
2007-10-06 00:17:44 +00:00
2007-08-26 15:16:40 +00:00
/* Open key */
error = reg_open_key_abs ( ctx , ctx , path , & tmp ) ;
2005-09-29 12:00:49 +00:00
2015-12-03 15:24:12 +01:00
if ( W_ERROR_EQUAL ( error , WERR_FILE_NOT_FOUND ) ) {
2007-08-26 15:16:40 +00:00
DEBUG ( 0 , ( " Error opening key '%s' \n " , path ) ) ;
return error ;
}
2005-09-29 12:00:49 +00:00
2007-08-26 15:16:40 +00:00
/* Set value */
2007-10-06 00:17:44 +00:00
error = reg_val_set ( tmp , value_name ,
2007-08-26 15:16:40 +00:00
value_type , value ) ;
if ( ! W_ERROR_IS_OK ( error ) ) {
DEBUG ( 0 , ( " Error setting value '%s' \n " , value_name ) ) ;
return error ;
2007-10-06 00:17:44 +00:00
}
2010-03-22 12:42:21 +01:00
talloc_free ( tmp ) ;
2007-08-26 15:16:40 +00:00
return WERR_OK ;
}
2005-09-29 12:00:49 +00:00
2007-10-06 00:17:44 +00:00
static WERROR reg_diff_apply_del_value ( void * _ctx , const char * key_name ,
const char * value_name )
2007-08-26 15:16:40 +00:00
{
2007-09-07 13:31:15 +00:00
struct registry_context * ctx = ( struct registry_context * ) _ctx ;
2007-08-26 15:16:40 +00:00
struct registry_key * tmp ;
WERROR error ;
2007-10-06 00:17:44 +00:00
2007-08-26 15:16:40 +00:00
/* Open key */
error = reg_open_key_abs ( ctx , ctx , key_name , & tmp ) ;
if ( ! W_ERROR_IS_OK ( error ) ) {
DEBUG ( 0 , ( " Error opening key '%s' \n " , key_name ) ) ;
return error ;
2005-09-29 12:00:49 +00:00
}
2010-03-22 19:18:56 +01:00
error = reg_del_value ( ctx , tmp , value_name ) ;
2007-08-26 15:16:40 +00:00
if ( ! W_ERROR_IS_OK ( error ) ) {
DEBUG ( 0 , ( " Error deleting value '%s' \n " , value_name ) ) ;
return error ;
}
2007-10-06 00:17:44 +00:00
2010-03-22 12:42:21 +01:00
talloc_free ( tmp ) ;
2005-09-29 12:00:49 +00:00
2007-08-26 15:16:40 +00:00
return WERR_OK ;
2005-09-29 12:00:49 +00:00
}
2007-08-26 15:16:40 +00:00
static WERROR reg_diff_apply_del_all_values ( void * _ctx , const char * key_name )
2005-09-29 12:00:49 +00:00
{
2007-09-07 13:31:15 +00:00
struct registry_context * ctx = ( struct registry_context * ) _ctx ;
2007-08-26 15:16:40 +00:00
struct registry_key * key ;
2005-09-29 12:00:49 +00:00
WERROR error ;
2010-03-22 12:42:21 +01:00
const char * value_name ;
2005-09-29 12:00:49 +00:00
2007-08-26 15:16:40 +00:00
error = reg_open_key_abs ( ctx , ctx , key_name , & key ) ;
2005-09-29 12:00:49 +00:00
2007-08-26 15:16:40 +00:00
if ( ! W_ERROR_IS_OK ( error ) ) {
DEBUG ( 0 , ( " Error opening key '%s' \n " , key_name ) ) ;
return error ;
}
2005-09-29 12:00:49 +00:00
2008-01-07 14:11:29 -06:00
W_ERROR_NOT_OK_RETURN ( reg_key_get_info ( ctx , key , NULL ,
2008-10-03 14:29:47 +02:00
NULL , NULL , NULL , NULL , NULL , NULL ) ) ;
2007-08-26 15:16:40 +00:00
2008-10-03 14:29:47 +02:00
while ( W_ERROR_IS_OK ( reg_key_get_value_by_index (
ctx , key , 0 , & value_name , NULL , NULL ) ) ) {
2010-03-22 19:18:56 +01:00
error = reg_del_value ( ctx , key , value_name ) ;
2008-10-03 14:29:47 +02:00
if ( ! W_ERROR_IS_OK ( error ) ) {
DEBUG ( 0 , ( " Error deleting value '%s' \n " , value_name ) ) ;
return error ;
}
2010-03-22 12:42:21 +01:00
talloc_free ( discard_const_p ( char , value_name ) ) ;
2007-08-26 15:16:40 +00:00
}
2005-09-29 12:00:49 +00:00
2010-03-22 12:42:21 +01:00
talloc_free ( key ) ;
2007-08-26 15:16:40 +00:00
return WERR_OK ;
}
2005-09-29 12:00:49 +00:00
2007-10-06 00:17:44 +00:00
/**
* Apply diff to a registry context
2007-08-26 15:16:40 +00:00
*/
2008-10-24 14:57:03 +02:00
_PUBLIC_ WERROR reg_diff_apply ( struct registry_context * ctx ,
const char * filename )
2007-08-26 15:16:40 +00:00
{
struct reg_diff_callbacks callbacks ;
2005-09-29 12:00:49 +00:00
2007-08-26 15:16:40 +00:00
callbacks . add_key = reg_diff_apply_add_key ;
callbacks . del_key = reg_diff_apply_del_key ;
callbacks . set_value = reg_diff_apply_set_value ;
callbacks . del_value = reg_diff_apply_del_value ;
callbacks . del_all_values = reg_diff_apply_del_all_values ;
callbacks . done = NULL ;
2005-09-29 12:00:49 +00:00
2010-05-09 17:20:01 +02:00
return reg_diff_load ( filename , & callbacks , ctx ) ;
2005-09-29 12:00:49 +00:00
}