2008-03-21 16:26:50 +01:00
/*
* Unix SMB / CIFS implementation .
* libsmbconf - Samba configuration library , registry backend
* Copyright ( C ) Michael Adam 2008
*
* 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 3 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 , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
2009-01-21 18:25:47 +01:00
# include "lib/smbconf/smbconf_private.h"
2009-10-02 00:17:06 +02:00
# include "registry.h"
2010-09-21 08:11:18 +02:00
# include "registry/reg_api.h"
2010-05-24 22:42:00 +02:00
# include "registry/reg_backend_db.h"
2010-05-24 23:37:09 +02:00
# include "registry/reg_util_token.h"
2010-09-20 02:42:10 +02:00
# include "registry/reg_api_util.h"
2010-09-21 08:50:54 +02:00
# include "registry/reg_init_smbconf.h"
2010-05-18 02:27:34 +02:00
# include "lib/smbconf/smbconf_init.h"
# include "lib/smbconf/smbconf_reg.h"
2011-02-26 00:28:15 +01:00
# include "../libcli/registry/util_reg.h"
2008-03-21 16:26:50 +01:00
2008-04-08 17:46:36 +02:00
# define INCLUDES_VALNAME "includes"
2008-03-21 22:52:27 +01:00
struct reg_private_data {
2009-02-21 18:00:42 +01:00
struct registry_key * base_key ;
2008-03-21 23:39:01 +01:00
bool open ; /* did _we_ open the registry? */
2008-03-21 22:52:27 +01:00
} ;
2008-03-21 16:26:50 +01:00
/**********************************************************************
*
* helper functions
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-03-21 22:55:20 +01:00
/**
* a convenience helper to cast the private data structure
*/
2008-03-21 22:52:27 +01:00
static struct reg_private_data * rpd ( struct smbconf_ctx * ctx )
{
return ( struct reg_private_data * ) ( ctx - > data ) ;
}
2013-09-22 03:39:48 +02:00
/**
* Check whether a given parameter name is valid in the
* smbconf registry backend .
2008-04-08 22:51:02 +02:00
*/
2013-09-22 10:27:38 +02:00
bool smbconf_reg_parameter_is_valid ( const char * param_name )
2008-04-08 22:51:02 +02:00
{
/* hard code the list of forbidden names here for now */
2013-09-22 03:39:48 +02:00
const char * forbidden_names [ ] = {
2013-09-17 19:10:48 +02:00
" state directory " ,
2008-04-08 22:51:02 +02:00
" lock directory " ,
" lock dir " ,
" config backend " ,
2008-04-09 01:19:50 +02:00
" include " ,
2013-09-22 08:54:30 +02:00
/*
* " includes " has a special meaning internally .
* It is currently not necessary to list it here since it is
* not a valid parameter . But for clarity and safety , we keep
* it for now .
*/
INCLUDES_VALNAME ,
2008-04-08 22:51:02 +02:00
NULL
} ;
const char * * forbidden = NULL ;
2013-09-22 03:39:48 +02:00
if ( ! lp_parameter_is_valid ( param_name ) ) {
return false ;
}
for ( forbidden = forbidden_names ; * forbidden ! = NULL ; forbidden + + ) {
if ( strwicmp ( param_name , * forbidden ) = = 0 ) {
return false ;
2008-04-08 22:51:02 +02:00
}
}
2013-09-22 03:39:48 +02:00
return true ;
2008-04-09 00:05:45 +02:00
}
2008-03-21 16:26:50 +01:00
/**
2008-03-21 17:55:31 +01:00
* Open a subkey of the base key ( i . e a service )
2008-03-21 16:26:50 +01:00
*/
2011-04-08 17:20:35 +02:00
static sbcErr smbconf_reg_open_service_key ( TALLOC_CTX * mem_ctx ,
2008-03-21 16:26:50 +01:00
struct smbconf_ctx * ctx ,
const char * servicename ,
uint32 desired_access ,
struct registry_key * * key )
{
2009-03-04 22:02:07 +01:00
WERROR werr ;
2008-03-21 16:26:50 +01:00
if ( servicename = = NULL ) {
2009-02-21 18:00:42 +01:00
* key = rpd ( ctx ) - > base_key ;
2011-04-08 17:20:35 +02:00
return SBC_ERR_OK ;
2008-03-21 16:26:50 +01:00
}
2009-03-04 22:02:07 +01:00
werr = reg_openkey ( mem_ctx , rpd ( ctx ) - > base_key , servicename ,
2009-02-21 18:00:42 +01:00
desired_access , key ) ;
2009-03-04 22:02:07 +01:00
if ( W_ERROR_EQUAL ( werr , WERR_BADFILE ) ) {
2011-04-08 17:20:35 +02:00
return SBC_ERR_NO_SUCH_SERVICE ;
}
if ( ! W_ERROR_IS_OK ( werr ) ) {
return SBC_ERR_NOMEM ;
2009-03-04 22:02:07 +01:00
}
2011-04-08 17:20:35 +02:00
return SBC_ERR_OK ;
2008-03-21 16:26:50 +01:00
}
/**
* check if a value exists in a given registry key
*/
static bool smbconf_value_exists ( struct registry_key * key , const char * param )
{
bool ret = false ;
WERROR werr = WERR_OK ;
TALLOC_CTX * ctx = talloc_stackframe ( ) ;
struct registry_value * value = NULL ;
werr = reg_queryvalue ( ctx , key , param , & value ) ;
if ( W_ERROR_IS_OK ( werr ) ) {
ret = true ;
}
2009-01-21 17:16:37 +01:00
talloc_free ( ctx ) ;
2008-03-21 16:26:50 +01:00
return ret ;
}
/**
2008-03-21 17:55:31 +01:00
* create a subkey of the base key ( i . e . a service . . . )
2008-03-21 16:26:50 +01:00
*/
2011-04-08 15:48:01 +02:00
static sbcErr smbconf_reg_create_service_key ( TALLOC_CTX * mem_ctx ,
2008-03-21 16:26:50 +01:00
struct smbconf_ctx * ctx ,
const char * subkeyname ,
struct registry_key * * newkey )
{
2011-04-08 15:48:01 +02:00
WERROR werr ;
sbcErr err = SBC_ERR_OK ;
2008-03-21 16:26:50 +01:00
TALLOC_CTX * create_ctx ;
enum winreg_CreateAction action = REG_ACTION_NONE ;
/* create a new talloc ctx for creation. it will hold
* the intermediate parent key ( SMBCONF ) for creation
* and will be destroyed when leaving this function . . . */
2008-04-23 01:48:26 +02:00
create_ctx = talloc_stackframe ( ) ;
2008-03-21 16:26:50 +01:00
2009-02-21 18:00:42 +01:00
werr = reg_createkey ( mem_ctx , rpd ( ctx ) - > base_key , subkeyname ,
2008-03-21 16:26:50 +01:00
REG_KEY_WRITE , newkey , & action ) ;
if ( W_ERROR_IS_OK ( werr ) & & ( action ! = REG_CREATED_NEW_KEY ) ) {
DEBUG ( 10 , ( " Key '%s' already exists. \n " , subkeyname ) ) ;
2011-04-08 15:48:01 +02:00
err = SBC_ERR_FILE_EXISTS ;
2008-03-21 16:26:50 +01:00
}
if ( ! W_ERROR_IS_OK ( werr ) ) {
DEBUG ( 5 , ( " Error creating key %s: %s \n " ,
2008-11-01 17:19:26 +01:00
subkeyname , win_errstr ( werr ) ) ) ;
2011-04-08 15:48:01 +02:00
err = SBC_ERR_UNKNOWN_FAILURE ;
2008-03-21 16:26:50 +01:00
}
2009-01-21 17:16:37 +01:00
talloc_free ( create_ctx ) ;
2011-04-08 15:48:01 +02:00
return err ;
2008-03-21 16:26:50 +01:00
}
/**
* add a value to a key .
*/
2011-04-11 13:23:26 +02:00
static sbcErr smbconf_reg_set_value ( struct registry_key * key ,
2008-03-21 16:26:50 +01:00
const char * valname ,
const char * valstr )
{
struct registry_value val ;
WERROR werr = WERR_OK ;
2011-04-11 13:23:26 +02:00
sbcErr err ;
2008-03-21 16:26:50 +01:00
char * subkeyname ;
const char * canon_valname ;
const char * canon_valstr ;
2013-09-22 08:47:14 +02:00
if ( ! lp_parameter_is_valid ( valname ) ) {
DEBUG ( 5 , ( " Invalid parameter '%s' given. \n " , valname ) ) ;
2011-04-11 13:23:26 +02:00
err = SBC_ERR_INVALID_PARAM ;
2008-03-21 16:26:50 +01:00
goto done ;
}
2013-09-22 08:47:14 +02:00
if ( ! smbconf_reg_parameter_is_valid ( valname ) ) {
2008-03-21 16:26:50 +01:00
DEBUG ( 5 , ( " Parameter '%s' not allowed in registry. \n " ,
2013-09-22 08:47:14 +02:00
valname ) ) ;
2011-04-11 13:23:26 +02:00
err = SBC_ERR_INVALID_PARAM ;
2008-03-21 16:26:50 +01:00
goto done ;
}
subkeyname = strrchr_m ( key - > key - > name , ' \\ ' ) ;
if ( ( subkeyname = = NULL ) | | ( * ( subkeyname + 1 ) = = ' \0 ' ) ) {
DEBUG ( 5 , ( " Invalid registry key '%s' given as "
" smbconf section. \n " , key - > key - > name ) ) ;
2011-04-11 13:23:26 +02:00
err = SBC_ERR_INVALID_PARAM ;
2008-03-21 16:26:50 +01:00
goto done ;
}
subkeyname + + ;
if ( ! strequal ( subkeyname , GLOBAL_NAME ) & &
lp_parameter_is_global ( valname ) )
{
2009-04-07 19:58:14 +02:00
DEBUG ( 5 , ( " Global parameter '%s' not allowed in "
2013-09-22 08:47:14 +02:00
" service definition ('%s'). \n " , valname ,
2008-03-21 16:26:50 +01:00
subkeyname ) ) ;
2011-04-11 13:23:26 +02:00
err = SBC_ERR_INVALID_PARAM ;
2008-03-21 16:26:50 +01:00
goto done ;
}
2013-09-22 08:47:14 +02:00
if ( ! lp_canonicalize_parameter_with_value ( valname , valstr ,
& canon_valname ,
& canon_valstr ) )
{
/*
* We already know the parameter name is valid .
* So the value must be invalid .
*/
DEBUG ( 5 , ( " invalid value '%s' given for parameter '%s' \n " ,
valstr , valname ) ) ;
err = SBC_ERR_INVALID_PARAM ;
goto done ;
}
2008-04-03 15:16:01 +02:00
ZERO_STRUCT ( val ) ;
val . type = REG_SZ ;
2010-06-29 16:13:15 +02:00
if ( ! push_reg_sz ( talloc_tos ( ) , & val . data , canon_valstr ) ) {
2011-04-11 13:23:26 +02:00
err = SBC_ERR_NOMEM ;
2010-06-29 16:13:15 +02:00
goto done ;
}
2008-04-03 15:16:01 +02:00
2008-03-21 16:26:50 +01:00
werr = reg_setvalue ( key , canon_valname , & val ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
DEBUG ( 5 , ( " Error adding value '%s' to "
" key '%s': %s \n " ,
2008-11-01 17:19:26 +01:00
canon_valname , key - > key - > name , win_errstr ( werr ) ) ) ;
2011-04-11 13:23:26 +02:00
err = SBC_ERR_NOMEM ;
goto done ;
2008-03-21 16:26:50 +01:00
}
2011-04-11 13:23:26 +02:00
err = SBC_ERR_OK ;
2008-03-21 16:26:50 +01:00
done :
2011-04-11 13:23:26 +02:00
return err ;
2008-03-21 16:26:50 +01:00
}
2011-04-11 15:14:52 +02:00
static sbcErr smbconf_reg_set_multi_sz_value ( struct registry_key * key ,
2008-04-08 17:46:36 +02:00
const char * valname ,
const uint32_t num_strings ,
const char * * strings )
{
WERROR werr ;
2011-04-11 15:14:52 +02:00
sbcErr err = SBC_ERR_OK ;
2008-04-08 17:46:36 +02:00
struct registry_value * value ;
uint32_t count ;
TALLOC_CTX * tmp_ctx = talloc_stackframe ( ) ;
2010-06-29 16:13:15 +02:00
const char * * array ;
2008-04-08 17:46:36 +02:00
if ( strings = = NULL ) {
2011-04-11 15:14:52 +02:00
err = SBC_ERR_INVALID_PARAM ;
2008-04-08 17:46:36 +02:00
goto done ;
}
2010-06-29 16:13:15 +02:00
array = talloc_zero_array ( tmp_ctx , const char * , num_strings + 1 ) ;
if ( array = = NULL ) {
2011-04-11 15:14:52 +02:00
err = SBC_ERR_NOMEM ;
2010-06-29 16:13:15 +02:00
goto done ;
}
2008-04-08 17:46:36 +02:00
2011-06-07 11:44:43 +10:00
value = talloc_zero ( tmp_ctx , struct registry_value ) ;
2010-06-29 16:13:15 +02:00
if ( value = = NULL ) {
2011-04-11 15:14:52 +02:00
err = SBC_ERR_NOMEM ;
2008-04-08 17:46:36 +02:00
goto done ;
}
2010-06-29 16:13:15 +02:00
value - > type = REG_MULTI_SZ ;
2008-04-08 17:46:36 +02:00
for ( count = 0 ; count < num_strings ; count + + ) {
2010-06-29 16:13:15 +02:00
array [ count ] = talloc_strdup ( value , strings [ count ] ) ;
if ( array [ count ] = = NULL ) {
2011-04-11 15:14:52 +02:00
err = SBC_ERR_NOMEM ;
2008-04-08 17:46:36 +02:00
goto done ;
}
}
2010-06-29 16:13:15 +02:00
if ( ! push_reg_multi_sz ( value , & value - > data , array ) ) {
2011-04-11 15:14:52 +02:00
err = SBC_ERR_NOMEM ;
2010-06-29 16:13:15 +02:00
goto done ;
}
2008-04-08 17:46:36 +02:00
werr = reg_setvalue ( key , valname , value ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
DEBUG ( 5 , ( " Error adding value '%s' to key '%s': %s \n " ,
2008-11-01 17:19:26 +01:00
valname , key - > key - > name , win_errstr ( werr ) ) ) ;
2011-04-11 15:14:52 +02:00
err = SBC_ERR_ACCESS_DENIED ;
2008-04-08 17:46:36 +02:00
}
done :
2009-01-21 17:16:37 +01:00
talloc_free ( tmp_ctx ) ;
2011-04-11 15:14:52 +02:00
return err ;
2008-04-08 17:46:36 +02:00
}
2008-03-21 16:26:50 +01:00
/**
* format a registry_value into a string .
*
* This is intended to be used for smbconf registry values ,
* which are ar stored as REG_SZ values , so the incomplete
* handling should be ok .
*/
static char * smbconf_format_registry_value ( TALLOC_CTX * mem_ctx ,
struct registry_value * value )
{
char * result = NULL ;
/* alternatively, create a new talloc context? */
if ( mem_ctx = = NULL ) {
return result ;
}
switch ( value - > type ) {
case REG_DWORD :
2010-06-29 16:13:15 +02:00
if ( value - > data . length > = 4 ) {
uint32_t v = IVAL ( value - > data . data , 0 ) ;
result = talloc_asprintf ( mem_ctx , " %d " , v ) ;
}
2008-03-21 16:26:50 +01:00
break ;
case REG_SZ :
2010-06-29 16:13:15 +02:00
case REG_EXPAND_SZ : {
const char * s ;
if ( ! pull_reg_sz ( mem_ctx , & value - > data , & s ) ) {
break ;
}
result = talloc_strdup ( mem_ctx , s ) ;
2008-03-21 16:26:50 +01:00
break ;
2010-06-29 16:13:15 +02:00
}
2008-03-21 16:26:50 +01:00
case REG_MULTI_SZ : {
2008-04-07 15:29:11 +02:00
uint32 j ;
2010-06-29 16:13:15 +02:00
const char * * a = NULL ;
if ( ! pull_reg_multi_sz ( mem_ctx , & value - > data , & a ) ) {
break ;
}
for ( j = 0 ; a [ j ] ! = NULL ; j + + ) {
2008-04-08 22:23:57 +02:00
result = talloc_asprintf ( mem_ctx , " %s \" %s \" " ,
result ? result : " " ,
2010-06-29 16:13:15 +02:00
a [ j ] ) ;
2008-03-21 16:26:50 +01:00
if ( result = = NULL ) {
break ;
}
2008-04-07 15:29:11 +02:00
}
break ;
}
2008-03-21 16:26:50 +01:00
case REG_BINARY :
2008-04-07 15:29:11 +02:00
result = talloc_asprintf ( mem_ctx , " binary (%d bytes) " ,
2010-06-29 16:13:15 +02:00
( int ) value - > data . length ) ;
2008-04-07 15:29:11 +02:00
break ;
default :
result = talloc_asprintf ( mem_ctx , " <unprintable> " ) ;
break ;
}
2008-03-21 16:26:50 +01:00
return result ;
}
2011-04-08 17:20:35 +02:00
static sbcErr smbconf_reg_get_includes_internal ( TALLOC_CTX * mem_ctx ,
2008-04-08 23:58:56 +02:00
struct registry_key * key ,
uint32_t * num_includes ,
char * * * includes )
{
WERROR werr ;
2011-04-08 14:19:15 +02:00
sbcErr err ;
2008-04-08 23:58:56 +02:00
uint32_t count ;
struct registry_value * value = NULL ;
char * * tmp_includes = NULL ;
2010-06-29 16:13:15 +02:00
const char * * array = NULL ;
2008-04-08 23:58:56 +02:00
TALLOC_CTX * tmp_ctx = talloc_stackframe ( ) ;
if ( ! smbconf_value_exists ( key , INCLUDES_VALNAME ) ) {
/* no includes */
2008-04-09 22:22:20 +02:00
* num_includes = 0 ;
* includes = NULL ;
2011-04-08 17:20:35 +02:00
err = SBC_ERR_OK ;
2008-04-08 23:58:56 +02:00
goto done ;
}
werr = reg_queryvalue ( tmp_ctx , key , INCLUDES_VALNAME , & value ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
2011-04-08 17:20:35 +02:00
err = SBC_ERR_ACCESS_DENIED ;
2008-04-08 23:58:56 +02:00
goto done ;
}
if ( value - > type ! = REG_MULTI_SZ ) {
2009-04-27 13:25:07 +02:00
/* wrong type -- ignore */
2011-04-08 17:20:35 +02:00
err = SBC_ERR_OK ;
2008-04-08 23:58:56 +02:00
goto done ;
}
2010-06-29 16:13:15 +02:00
if ( ! pull_reg_multi_sz ( tmp_ctx , & value - > data , & array ) ) {
2011-04-08 17:20:35 +02:00
err = SBC_ERR_NOMEM ;
2010-06-29 16:13:15 +02:00
goto done ;
}
for ( count = 0 ; array [ count ] ! = NULL ; count + + ) {
2011-04-08 14:19:15 +02:00
err = smbconf_add_string_to_array ( tmp_ctx ,
2008-04-08 23:58:56 +02:00
& tmp_includes ,
count ,
2010-06-29 16:13:15 +02:00
array [ count ] ) ;
2011-04-08 14:19:15 +02:00
if ( ! SBC_ERROR_IS_OK ( err ) ) {
2008-04-08 23:58:56 +02:00
goto done ;
}
}
if ( count > 0 ) {
* includes = talloc_move ( mem_ctx , & tmp_includes ) ;
if ( * includes = = NULL ) {
2011-04-08 17:20:35 +02:00
err = SBC_ERR_NOMEM ;
2008-04-08 23:58:56 +02:00
goto done ;
}
* num_includes = count ;
} else {
* num_includes = 0 ;
* includes = NULL ;
}
2011-04-11 14:52:52 +02:00
err = SBC_ERR_OK ;
2008-04-08 23:58:56 +02:00
done :
2009-01-21 17:16:37 +01:00
talloc_free ( tmp_ctx ) ;
2011-04-08 17:20:35 +02:00
return err ;
2008-04-08 23:58:56 +02:00
}
2008-03-21 16:26:50 +01:00
/**
* Get the values of a key as a list of value names
* and a list of value strings ( ordered )
*/
2011-04-08 17:20:35 +02:00
static sbcErr smbconf_reg_get_values ( TALLOC_CTX * mem_ctx ,
2008-03-21 16:26:50 +01:00
struct registry_key * key ,
uint32_t * num_values ,
char * * * value_names ,
char * * * value_strings )
{
TALLOC_CTX * tmp_ctx = NULL ;
WERROR werr = WERR_OK ;
2011-04-08 14:19:15 +02:00
sbcErr err ;
2008-03-21 16:26:50 +01:00
uint32_t count ;
struct registry_value * valvalue = NULL ;
char * valname = NULL ;
2008-04-09 00:05:45 +02:00
uint32_t tmp_num_values = 0 ;
2008-03-21 16:26:50 +01:00
char * * tmp_valnames = NULL ;
char * * tmp_valstrings = NULL ;
2008-04-09 00:47:27 +02:00
uint32_t num_includes = 0 ;
char * * includes = NULL ;
2008-03-21 16:26:50 +01:00
if ( ( num_values = = NULL ) | | ( value_names = = NULL ) | |
( value_strings = = NULL ) )
{
2011-04-08 17:20:35 +02:00
err = SBC_ERR_INVALID_PARAM ;
2008-03-21 16:26:50 +01:00
goto done ;
}
tmp_ctx = talloc_stackframe ( ) ;
for ( count = 0 ;
2008-04-07 15:15:57 +02:00
werr = reg_enumvalue ( tmp_ctx , key , count , & valname , & valvalue ) ,
W_ERROR_IS_OK ( werr ) ;
2008-03-21 16:26:50 +01:00
count + + )
{
char * valstring ;
2013-09-22 03:39:48 +02:00
if ( ! smbconf_reg_parameter_is_valid ( valname ) ) {
2008-04-09 00:05:45 +02:00
continue ;
}
2011-04-08 14:19:15 +02:00
err = smbconf_add_string_to_array ( tmp_ctx ,
& tmp_valnames ,
tmp_num_values , valname ) ;
if ( ! SBC_ERROR_IS_OK ( err ) ) {
2008-03-21 16:26:50 +01:00
goto done ;
}
valstring = smbconf_format_registry_value ( tmp_ctx , valvalue ) ;
2011-04-08 14:19:15 +02:00
err = smbconf_add_string_to_array ( tmp_ctx , & tmp_valstrings ,
tmp_num_values , valstring ) ;
if ( ! SBC_ERROR_IS_OK ( err ) ) {
2008-03-21 16:26:50 +01:00
goto done ;
}
2008-04-09 00:05:45 +02:00
tmp_num_values + + ;
2008-03-21 16:26:50 +01:00
}
if ( ! W_ERROR_EQUAL ( WERR_NO_MORE_ITEMS , werr ) ) {
2011-04-08 17:20:35 +02:00
err = SBC_ERR_NOMEM ;
2008-03-21 16:26:50 +01:00
goto done ;
}
2008-04-09 00:47:27 +02:00
/* now add the includes at the end */
2011-04-08 17:20:35 +02:00
err = smbconf_reg_get_includes_internal ( tmp_ctx , key , & num_includes ,
2008-04-09 00:47:27 +02:00
& includes ) ;
2011-04-08 17:20:35 +02:00
if ( ! SBC_ERROR_IS_OK ( err ) ) {
2008-04-09 00:47:27 +02:00
goto done ;
}
2011-04-08 17:20:35 +02:00
2008-04-09 00:47:27 +02:00
for ( count = 0 ; count < num_includes ; count + + ) {
2011-04-08 14:19:15 +02:00
err = smbconf_add_string_to_array ( tmp_ctx , & tmp_valnames ,
tmp_num_values , " include " ) ;
if ( ! SBC_ERROR_IS_OK ( err ) ) {
2008-04-09 00:47:27 +02:00
goto done ;
}
2011-04-08 14:19:15 +02:00
err = smbconf_add_string_to_array ( tmp_ctx , & tmp_valstrings ,
tmp_num_values ,
includes [ count ] ) ;
if ( ! SBC_ERROR_IS_OK ( err ) ) {
2008-04-09 00:47:27 +02:00
goto done ;
}
tmp_num_values + + ;
}
2008-03-21 16:26:50 +01:00
2008-04-09 00:05:45 +02:00
* num_values = tmp_num_values ;
if ( tmp_num_values > 0 ) {
2008-03-21 16:26:50 +01:00
* value_names = talloc_move ( mem_ctx , & tmp_valnames ) ;
* value_strings = talloc_move ( mem_ctx , & tmp_valstrings ) ;
} else {
* value_names = NULL ;
* value_strings = NULL ;
}
done :
2009-01-21 17:16:37 +01:00
talloc_free ( tmp_ctx ) ;
2011-04-08 17:20:35 +02:00
return err ;
2008-03-21 16:26:50 +01:00
}
2008-04-15 17:39:01 +02:00
/**
* delete all values from a key
*/
2011-04-11 11:39:03 +02:00
static sbcErr smbconf_reg_delete_values ( struct registry_key * key )
2008-04-15 17:39:01 +02:00
{
WERROR werr ;
2011-04-11 11:39:03 +02:00
sbcErr err ;
2008-04-15 17:39:01 +02:00
char * valname ;
struct registry_value * valvalue ;
uint32_t count ;
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
for ( count = 0 ;
werr = reg_enumvalue ( mem_ctx , key , count , & valname , & valvalue ) ,
W_ERROR_IS_OK ( werr ) ;
count + + )
{
werr = reg_deletevalue ( key , valname ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
2011-04-11 11:39:03 +02:00
err = SBC_ERR_ACCESS_DENIED ;
2008-04-15 17:39:01 +02:00
goto done ;
}
}
if ( ! W_ERROR_EQUAL ( WERR_NO_MORE_ITEMS , werr ) ) {
DEBUG ( 1 , ( " smbconf_reg_delete_values: "
" Error enumerating values of %s: %s \n " ,
key - > key - > name ,
2008-11-01 17:19:26 +01:00
win_errstr ( werr ) ) ) ;
2011-04-11 11:39:03 +02:00
err = SBC_ERR_ACCESS_DENIED ;
2008-04-15 17:39:01 +02:00
goto done ;
}
2011-04-11 11:39:03 +02:00
err = SBC_ERR_OK ;
2008-04-15 17:39:01 +02:00
done :
2009-01-21 17:16:37 +01:00
talloc_free ( mem_ctx ) ;
2011-04-11 11:39:03 +02:00
return err ;
2008-04-15 17:39:01 +02:00
}
2008-03-21 16:26:50 +01:00
/**********************************************************************
*
* smbconf operations : registry implementations
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/**
* initialize the registry smbconf backend
*/
2011-04-08 10:28:17 +02:00
static sbcErr smbconf_reg_init ( struct smbconf_ctx * ctx , const char * path )
2008-03-21 16:26:50 +01:00
{
WERROR werr = WERR_OK ;
2011-04-08 10:28:17 +02:00
sbcErr err ;
2010-08-26 20:04:11 +10:00
struct security_token * token ;
2008-03-21 16:26:50 +01:00
2008-03-21 17:55:31 +01:00
if ( path = = NULL ) {
path = KEY_SMBCONF ;
}
ctx - > path = talloc_strdup ( ctx , path ) ;
if ( ctx - > path = = NULL ) {
2011-05-12 17:03:20 +02:00
err = SBC_ERR_NOMEM ;
2008-03-21 17:55:31 +01:00
goto done ;
}
2011-06-07 11:44:43 +10:00
ctx - > data = talloc_zero ( ctx , struct reg_private_data ) ;
2008-03-21 16:26:50 +01:00
2009-02-21 18:00:42 +01:00
werr = ntstatus_to_werror ( registry_create_admin_token ( ctx , & token ) ) ;
2008-03-21 16:26:50 +01:00
if ( ! W_ERROR_IS_OK ( werr ) ) {
DEBUG ( 1 , ( " Error creating admin token \n " ) ) ;
2011-04-08 10:28:17 +02:00
err = SBC_ERR_UNKNOWN_FAILURE ;
2008-03-21 16:26:50 +01:00
goto done ;
}
2008-03-21 23:39:01 +01:00
rpd ( ctx ) - > open = false ;
2008-03-21 16:26:50 +01:00
2008-04-13 15:25:47 +02:00
werr = registry_init_smbconf ( path ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
2011-04-08 10:28:17 +02:00
err = SBC_ERR_BADFILE ;
2008-03-21 22:52:27 +01:00
goto done ;
}
2011-04-08 10:28:17 +02:00
err = ctx - > ops - > open_conf ( ctx ) ;
if ( ! SBC_ERROR_IS_OK ( err ) ) {
2009-02-21 18:00:42 +01:00
DEBUG ( 1 , ( " Error opening the registry. \n " ) ) ;
goto done ;
}
werr = reg_open_path ( ctx , ctx - > path ,
2009-04-15 01:30:12 +02:00
KEY_ENUMERATE_SUB_KEYS | REG_KEY_WRITE ,
2009-02-21 18:00:42 +01:00
token , & rpd ( ctx ) - > base_key ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
2011-04-08 10:28:17 +02:00
err = SBC_ERR_UNKNOWN_FAILURE ;
2009-02-21 18:00:42 +01:00
goto done ;
}
2008-03-21 16:26:50 +01:00
done :
2011-04-08 10:28:17 +02:00
return err ;
2008-03-21 16:26:50 +01:00
}
static int smbconf_reg_shutdown ( struct smbconf_ctx * ctx )
{
2008-03-21 23:39:01 +01:00
return ctx - > ops - > close_conf ( ctx ) ;
2008-03-21 16:26:50 +01:00
}
2008-10-20 23:52:02 +02:00
static bool smbconf_reg_requires_messaging ( struct smbconf_ctx * ctx )
{
if ( lp_clustering ( ) & & lp_parm_bool ( - 1 , " ctdb " , " registry.tdb " , true ) ) {
return true ;
}
2014-01-21 14:34:48 +01:00
2008-10-20 23:52:02 +02:00
return false ;
}
2008-10-23 11:16:50 +02:00
static bool smbconf_reg_is_writeable ( struct smbconf_ctx * ctx )
{
/*
* The backend has write support .
*
* TODO : add access checks whether the concrete
* config source is really writeable by the calling user .
*/
return true ;
}
2011-04-08 10:28:17 +02:00
static sbcErr smbconf_reg_open ( struct smbconf_ctx * ctx )
2008-03-21 16:26:50 +01:00
{
2008-03-21 23:39:01 +01:00
WERROR werr ;
if ( rpd ( ctx ) - > open ) {
2011-04-08 10:28:17 +02:00
return SBC_ERR_OK ;
2008-03-21 23:39:01 +01:00
}
werr = regdb_open ( ) ;
2011-04-08 10:28:17 +02:00
if ( ! W_ERROR_IS_OK ( werr ) ) {
return SBC_ERR_BADFILE ;
2008-03-21 23:39:01 +01:00
}
2011-04-08 10:28:17 +02:00
rpd ( ctx ) - > open = true ;
return SBC_ERR_OK ;
2008-03-21 16:26:50 +01:00
}
static int smbconf_reg_close ( struct smbconf_ctx * ctx )
{
2008-03-21 23:39:01 +01:00
int ret ;
if ( ! rpd ( ctx ) - > open ) {
return 0 ;
}
ret = regdb_close ( ) ;
if ( ret = = 0 ) {
rpd ( ctx ) - > open = false ;
}
return ret ;
2008-03-21 16:26:50 +01:00
}
/**
* Get the change sequence number of the given service / parameter .
* service and parameter strings may be NULL .
*/
static void smbconf_reg_get_csn ( struct smbconf_ctx * ctx ,
struct smbconf_csn * csn ,
const char * service , const char * param )
{
if ( csn = = NULL ) {
return ;
}
2008-03-21 23:39:01 +01:00
2011-04-08 10:28:17 +02:00
if ( ! SBC_ERROR_IS_OK ( ctx - > ops - > open_conf ( ctx ) ) ) {
2008-03-21 23:39:01 +01:00
return ;
}
2008-03-21 16:26:50 +01:00
csn - > csn = ( uint64_t ) regdb_get_seqnum ( ) ;
}
/**
* Drop the whole configuration ( restarting empty ) - registry version
*/
2011-04-08 10:40:02 +02:00
static sbcErr smbconf_reg_drop ( struct smbconf_ctx * ctx )
2008-03-21 16:26:50 +01:00
{
char * path , * p ;
WERROR werr = WERR_OK ;
2011-04-08 10:40:02 +02:00
sbcErr err = SBC_ERR_OK ;
2008-03-21 16:26:50 +01:00
struct registry_key * parent_key = NULL ;
struct registry_key * new_key = NULL ;
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
enum winreg_CreateAction action ;
2010-08-26 20:04:11 +10:00
struct security_token * token ;
2009-02-21 18:00:42 +01:00
werr = ntstatus_to_werror ( registry_create_admin_token ( ctx , & token ) ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
DEBUG ( 1 , ( " Error creating admin token \n " ) ) ;
2011-04-08 10:40:02 +02:00
err = SBC_ERR_UNKNOWN_FAILURE ;
2009-02-21 18:00:42 +01:00
goto done ;
}
2008-03-21 16:26:50 +01:00
2008-03-21 17:55:31 +01:00
path = talloc_strdup ( mem_ctx , ctx - > path ) ;
2008-03-21 16:26:50 +01:00
if ( path = = NULL ) {
2011-04-08 10:40:02 +02:00
err = SBC_ERR_NOMEM ;
2008-03-21 16:26:50 +01:00
goto done ;
}
p = strrchr ( path , ' \\ ' ) ;
2011-04-21 22:09:27 +02:00
if ( p = = NULL ) {
2011-04-08 10:40:02 +02:00
err = SBC_ERR_INVALID_PARAM ;
2011-04-21 22:09:27 +02:00
goto done ;
}
2008-03-21 16:26:50 +01:00
* p = ' \0 ' ;
2009-02-21 18:00:42 +01:00
werr = reg_open_path ( mem_ctx , path , REG_KEY_WRITE , token ,
& parent_key ) ;
2008-03-21 16:26:50 +01:00
if ( ! W_ERROR_IS_OK ( werr ) ) {
2011-04-08 10:40:02 +02:00
err = SBC_ERR_IO_FAILURE ;
2008-03-21 16:26:50 +01:00
goto done ;
}
2011-08-02 14:52:12 +02:00
werr = reg_deletesubkeys_recursive ( parent_key , p + 1 ) ;
2008-03-21 16:26:50 +01:00
if ( ! W_ERROR_IS_OK ( werr ) ) {
2011-04-08 10:40:02 +02:00
err = SBC_ERR_IO_FAILURE ;
2008-03-21 16:26:50 +01:00
goto done ;
}
werr = reg_createkey ( mem_ctx , parent_key , p + 1 , REG_KEY_WRITE ,
& new_key , & action ) ;
2011-04-08 10:40:02 +02:00
if ( ! W_ERROR_IS_OK ( werr ) ) {
err = SBC_ERR_IO_FAILURE ;
goto done ;
}
2008-03-21 16:26:50 +01:00
done :
2009-01-21 17:16:37 +01:00
talloc_free ( mem_ctx ) ;
2011-04-08 10:40:02 +02:00
return err ;
2008-03-21 16:26:50 +01:00
}
/**
* get the list of share names defined in the configuration .
* registry version .
*/
2011-04-08 14:19:15 +02:00
static sbcErr smbconf_reg_get_share_names ( struct smbconf_ctx * ctx ,
2008-03-21 16:26:50 +01:00
TALLOC_CTX * mem_ctx ,
uint32_t * num_shares ,
char * * * share_names )
{
uint32_t count ;
uint32_t added_count = 0 ;
TALLOC_CTX * tmp_ctx = NULL ;
2011-04-08 14:19:15 +02:00
WERROR werr ;
sbcErr err = SBC_ERR_OK ;
2008-03-21 16:26:50 +01:00
char * subkey_name = NULL ;
char * * tmp_share_names = NULL ;
if ( ( num_shares = = NULL ) | | ( share_names = = NULL ) ) {
2011-04-08 14:19:15 +02:00
return SBC_ERR_INVALID_PARAM ;
2008-03-21 16:26:50 +01:00
}
tmp_ctx = talloc_stackframe ( ) ;
2008-04-15 17:39:01 +02:00
/* make sure "global" is always listed first */
if ( smbconf_share_exists ( ctx , GLOBAL_NAME ) ) {
2011-04-08 14:19:15 +02:00
err = smbconf_add_string_to_array ( tmp_ctx , & tmp_share_names ,
added_count , GLOBAL_NAME ) ;
if ( ! SBC_ERROR_IS_OK ( err ) ) {
2008-04-15 17:39:01 +02:00
goto done ;
}
added_count + + ;
2008-03-21 16:26:50 +01:00
}
for ( count = 0 ;
2009-02-21 18:00:42 +01:00
werr = reg_enumkey ( tmp_ctx , rpd ( ctx ) - > base_key , count ,
& subkey_name , NULL ) ,
2008-04-07 15:15:57 +02:00
W_ERROR_IS_OK ( werr ) ;
2008-03-21 16:26:50 +01:00
count + + )
{
if ( strequal ( subkey_name , GLOBAL_NAME ) ) {
continue ;
}
2011-04-08 14:19:15 +02:00
err = smbconf_add_string_to_array ( tmp_ctx ,
2008-03-21 16:26:50 +01:00
& tmp_share_names ,
added_count ,
subkey_name ) ;
2011-04-08 14:19:15 +02:00
if ( ! SBC_ERROR_IS_OK ( err ) ) {
2008-03-21 16:26:50 +01:00
goto done ;
}
added_count + + ;
}
if ( ! W_ERROR_EQUAL ( WERR_NO_MORE_ITEMS , werr ) ) {
2011-04-08 14:19:15 +02:00
err = SBC_ERR_NO_MORE_ITEMS ;
2008-03-21 16:26:50 +01:00
goto done ;
}
2011-04-08 14:19:15 +02:00
err = SBC_ERR_OK ;
2008-03-21 16:26:50 +01:00
* num_shares = added_count ;
if ( added_count > 0 ) {
* share_names = talloc_move ( mem_ctx , & tmp_share_names ) ;
} else {
* share_names = NULL ;
}
done :
2009-01-21 17:16:37 +01:00
talloc_free ( tmp_ctx ) ;
2011-04-08 14:19:15 +02:00
return err ;
2008-03-21 16:26:50 +01:00
}
/**
* check if a share / service of a given name exists - registry version
*/
static bool smbconf_reg_share_exists ( struct smbconf_ctx * ctx ,
const char * servicename )
{
bool ret = false ;
2011-04-08 17:20:35 +02:00
sbcErr err ;
2008-03-21 16:26:50 +01:00
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
struct registry_key * key = NULL ;
2011-04-08 17:20:35 +02:00
err = smbconf_reg_open_service_key ( mem_ctx , ctx , servicename ,
REG_KEY_READ , & key ) ;
if ( SBC_ERROR_IS_OK ( err ) ) {
2008-03-21 16:26:50 +01:00
ret = true ;
}
2009-01-21 17:16:37 +01:00
talloc_free ( mem_ctx ) ;
2008-03-21 16:26:50 +01:00
return ret ;
}
/**
* Add a service if it does not already exist - registry version
*/
2011-04-08 15:48:01 +02:00
static sbcErr smbconf_reg_create_share ( struct smbconf_ctx * ctx ,
2008-03-21 16:26:50 +01:00
const char * servicename )
{
2011-04-08 15:48:01 +02:00
sbcErr err ;
2008-03-21 16:26:50 +01:00
struct registry_key * key = NULL ;
2008-04-15 17:39:01 +02:00
if ( servicename = = NULL ) {
2011-04-08 15:48:01 +02:00
return SBC_ERR_OK ;
2008-04-15 17:39:01 +02:00
}
2008-03-21 16:26:50 +01:00
2011-04-08 15:48:01 +02:00
err = smbconf_reg_create_service_key ( talloc_tos ( ) , ctx ,
servicename , & key ) ;
2009-02-21 18:00:42 +01:00
talloc_free ( key ) ;
2011-04-08 15:48:01 +02:00
return err ;
2008-03-21 16:26:50 +01:00
}
/**
* get a definition of a share ( service ) from configuration .
*/
2011-04-08 17:20:35 +02:00
static sbcErr smbconf_reg_get_share ( struct smbconf_ctx * ctx ,
2008-03-21 16:26:50 +01:00
TALLOC_CTX * mem_ctx ,
const char * servicename ,
2008-04-22 16:31:16 +02:00
struct smbconf_service * * service )
2008-03-21 16:26:50 +01:00
{
2011-04-08 17:20:35 +02:00
sbcErr err ;
2008-03-21 16:26:50 +01:00
struct registry_key * key = NULL ;
2008-04-22 16:31:16 +02:00
struct smbconf_service * tmp_service = NULL ;
TALLOC_CTX * tmp_ctx = talloc_stackframe ( ) ;
2008-03-21 16:26:50 +01:00
2011-04-08 17:20:35 +02:00
err = smbconf_reg_open_service_key ( tmp_ctx , ctx , servicename ,
REG_KEY_READ , & key ) ;
if ( ! SBC_ERROR_IS_OK ( err ) ) {
2008-03-21 16:26:50 +01:00
goto done ;
}
2011-06-07 11:44:43 +10:00
tmp_service = talloc_zero ( tmp_ctx , struct smbconf_service ) ;
2008-04-22 16:31:16 +02:00
if ( tmp_service = = NULL ) {
2011-04-08 17:20:35 +02:00
err = SBC_ERR_NOMEM ;
2008-04-22 16:31:16 +02:00
goto done ;
}
if ( servicename ! = NULL ) {
tmp_service - > name = talloc_strdup ( tmp_service , servicename ) ;
if ( tmp_service - > name = = NULL ) {
2011-04-08 17:20:35 +02:00
err = SBC_ERR_NOMEM ;
2008-04-22 16:31:16 +02:00
goto done ;
}
}
2011-04-08 17:20:35 +02:00
err = smbconf_reg_get_values ( tmp_service , key ,
& ( tmp_service - > num_params ) ,
& ( tmp_service - > param_names ) ,
& ( tmp_service - > param_values ) ) ;
if ( SBC_ERROR_IS_OK ( err ) ) {
2008-04-22 16:31:16 +02:00
* service = talloc_move ( mem_ctx , & tmp_service ) ;
}
2008-03-21 16:26:50 +01:00
done :
2009-01-21 17:16:37 +01:00
talloc_free ( tmp_ctx ) ;
2011-04-08 17:20:35 +02:00
return err ;
2008-03-21 16:26:50 +01:00
}
/**
* delete a service from configuration
*/
2011-04-11 11:39:03 +02:00
static sbcErr smbconf_reg_delete_share ( struct smbconf_ctx * ctx ,
2008-03-21 16:26:50 +01:00
const char * servicename )
{
2011-04-11 11:39:03 +02:00
WERROR werr ;
sbcErr err = SBC_ERR_OK ;
2008-03-21 16:26:50 +01:00
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
2008-04-15 17:39:01 +02:00
if ( servicename ! = NULL ) {
2010-09-22 06:21:38 +02:00
werr = reg_deletekey_recursive ( rpd ( ctx ) - > base_key , servicename ) ;
2011-04-11 11:39:03 +02:00
if ( ! W_ERROR_IS_OK ( werr ) ) {
err = SBC_ERR_ACCESS_DENIED ;
}
2008-04-15 17:39:01 +02:00
} else {
2011-04-11 11:39:03 +02:00
err = smbconf_reg_delete_values ( rpd ( ctx ) - > base_key ) ;
2008-04-15 17:39:01 +02:00
}
2008-03-21 16:26:50 +01:00
2009-01-21 17:16:37 +01:00
talloc_free ( mem_ctx ) ;
2011-04-11 11:39:03 +02:00
return err ;
2008-03-21 16:26:50 +01:00
}
/**
* set a configuration parameter to the value provided .
*/
2011-04-11 13:23:26 +02:00
static sbcErr smbconf_reg_set_parameter ( struct smbconf_ctx * ctx ,
2008-03-21 16:26:50 +01:00
const char * service ,
const char * param ,
const char * valstr )
{
2011-04-08 17:20:35 +02:00
sbcErr err ;
2008-03-21 16:26:50 +01:00
struct registry_key * key = NULL ;
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
2011-04-08 17:20:35 +02:00
err = smbconf_reg_open_service_key ( mem_ctx , ctx , service ,
REG_KEY_WRITE , & key ) ;
if ( ! SBC_ERROR_IS_OK ( err ) ) {
2008-03-21 16:26:50 +01:00
goto done ;
}
2011-04-11 13:23:26 +02:00
err = smbconf_reg_set_value ( key , param , valstr ) ;
2008-03-21 16:26:50 +01:00
done :
2009-01-21 17:16:37 +01:00
talloc_free ( mem_ctx ) ;
2011-04-11 13:23:26 +02:00
return err ;
2008-03-21 16:26:50 +01:00
}
/**
* get the value of a configuration parameter as a string
*/
2011-04-11 13:50:53 +02:00
static sbcErr smbconf_reg_get_parameter ( struct smbconf_ctx * ctx ,
2008-03-21 16:26:50 +01:00
TALLOC_CTX * mem_ctx ,
const char * service ,
const char * param ,
char * * valstr )
{
WERROR werr = WERR_OK ;
2011-04-08 17:20:35 +02:00
sbcErr err ;
2008-03-21 16:26:50 +01:00
struct registry_key * key = NULL ;
struct registry_value * value = NULL ;
2011-04-08 17:20:35 +02:00
err = smbconf_reg_open_service_key ( mem_ctx , ctx , service ,
REG_KEY_READ , & key ) ;
if ( ! SBC_ERROR_IS_OK ( err ) ) {
2008-03-21 16:26:50 +01:00
goto done ;
}
2013-09-22 03:39:48 +02:00
if ( ! smbconf_reg_parameter_is_valid ( param ) ) {
2011-04-11 13:50:53 +02:00
err = SBC_ERR_INVALID_PARAM ;
2008-04-09 09:54:17 +02:00
goto done ;
}
2008-03-21 16:26:50 +01:00
if ( ! smbconf_value_exists ( key , param ) ) {
2011-04-11 13:50:53 +02:00
err = SBC_ERR_INVALID_PARAM ;
2008-03-21 16:26:50 +01:00
goto done ;
}
werr = reg_queryvalue ( mem_ctx , key , param , & value ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
2011-04-11 13:50:53 +02:00
err = SBC_ERR_NOMEM ;
2008-03-21 16:26:50 +01:00
goto done ;
}
* valstr = smbconf_format_registry_value ( mem_ctx , value ) ;
if ( * valstr = = NULL ) {
2011-04-11 13:50:53 +02:00
err = SBC_ERR_NOMEM ;
2008-03-21 16:26:50 +01:00
}
done :
2009-01-21 17:16:37 +01:00
talloc_free ( key ) ;
talloc_free ( value ) ;
2011-04-11 13:50:53 +02:00
return err ;
2008-03-21 16:26:50 +01:00
}
/**
* delete a parameter from configuration
*/
2011-04-11 14:20:32 +02:00
static sbcErr smbconf_reg_delete_parameter ( struct smbconf_ctx * ctx ,
2008-03-21 16:26:50 +01:00
const char * service ,
const char * param )
{
struct registry_key * key = NULL ;
2011-04-11 14:20:32 +02:00
WERROR werr ;
2011-04-08 17:20:35 +02:00
sbcErr err ;
2008-03-21 16:26:50 +01:00
TALLOC_CTX * mem_ctx = talloc_stackframe ( ) ;
2011-04-08 17:20:35 +02:00
err = smbconf_reg_open_service_key ( mem_ctx , ctx , service ,
REG_KEY_ALL , & key ) ;
if ( ! SBC_ERROR_IS_OK ( err ) ) {
2008-03-21 16:26:50 +01:00
goto done ;
}
2013-09-22 03:39:48 +02:00
if ( ! smbconf_reg_parameter_is_valid ( param ) ) {
2011-04-11 14:20:32 +02:00
err = SBC_ERR_INVALID_PARAM ;
2008-04-09 09:54:17 +02:00
goto done ;
}
2008-03-21 16:26:50 +01:00
if ( ! smbconf_value_exists ( key , param ) ) {
2011-04-11 16:01:22 +02:00
err = SBC_ERR_OK ;
2008-03-21 16:26:50 +01:00
goto done ;
}
werr = reg_deletevalue ( key , param ) ;
2011-04-11 14:20:32 +02:00
if ( ! W_ERROR_IS_OK ( werr ) ) {
err = SBC_ERR_ACCESS_DENIED ;
}
2008-03-21 16:26:50 +01:00
done :
2009-01-21 17:16:37 +01:00
talloc_free ( mem_ctx ) ;
2011-04-11 14:20:32 +02:00
return err ;
2008-03-21 16:26:50 +01:00
}
2011-04-11 14:52:52 +02:00
static sbcErr smbconf_reg_get_includes ( struct smbconf_ctx * ctx ,
2008-04-08 10:16:03 +02:00
TALLOC_CTX * mem_ctx ,
2008-04-08 01:56:32 +02:00
const char * service ,
uint32_t * num_includes ,
char * * * includes )
{
2011-04-08 17:20:35 +02:00
sbcErr err ;
2008-04-08 17:46:36 +02:00
struct registry_key * key = NULL ;
TALLOC_CTX * tmp_ctx = talloc_stackframe ( ) ;
2011-04-08 17:20:35 +02:00
err = smbconf_reg_open_service_key ( tmp_ctx , ctx , service ,
REG_KEY_READ , & key ) ;
if ( ! SBC_ERROR_IS_OK ( err ) ) {
2008-04-08 17:46:36 +02:00
goto done ;
}
2011-04-08 17:20:35 +02:00
err = smbconf_reg_get_includes_internal ( mem_ctx , key , num_includes ,
2008-04-08 23:58:56 +02:00
includes ) ;
2011-04-08 17:20:35 +02:00
if ( ! SBC_ERROR_IS_OK ( err ) ) {
goto done ;
}
2008-04-08 17:46:36 +02:00
done :
2009-01-21 17:16:37 +01:00
talloc_free ( tmp_ctx ) ;
2011-04-11 14:52:52 +02:00
return err ;
2008-04-08 01:56:32 +02:00
}
2011-04-11 15:14:52 +02:00
static sbcErr smbconf_reg_set_includes ( struct smbconf_ctx * ctx ,
2008-04-08 01:56:32 +02:00
const char * service ,
uint32_t num_includes ,
const char * * includes )
{
2008-04-08 17:46:36 +02:00
WERROR werr = WERR_OK ;
2011-04-08 17:20:35 +02:00
sbcErr err ;
2008-04-08 17:46:36 +02:00
struct registry_key * key = NULL ;
TALLOC_CTX * tmp_ctx = talloc_stackframe ( ) ;
2011-04-08 17:20:35 +02:00
err = smbconf_reg_open_service_key ( tmp_ctx , ctx , service ,
REG_KEY_ALL , & key ) ;
if ( ! SBC_ERROR_IS_OK ( err ) ) {
2008-04-08 17:46:36 +02:00
goto done ;
}
2008-04-09 18:46:02 +02:00
if ( num_includes = = 0 ) {
if ( ! smbconf_value_exists ( key , INCLUDES_VALNAME ) ) {
2011-04-11 15:14:52 +02:00
err = SBC_ERR_OK ;
2008-04-09 18:46:02 +02:00
goto done ;
}
werr = reg_deletevalue ( key , INCLUDES_VALNAME ) ;
2011-04-11 15:14:52 +02:00
if ( ! W_ERROR_IS_OK ( werr ) ) {
err = SBC_ERR_ACCESS_DENIED ;
goto done ;
}
2008-04-09 18:46:02 +02:00
} else {
2011-04-11 15:14:52 +02:00
err = smbconf_reg_set_multi_sz_value ( key , INCLUDES_VALNAME ,
2008-04-09 18:46:02 +02:00
num_includes , includes ) ;
}
2008-04-08 17:46:36 +02:00
done :
2009-01-21 17:16:37 +01:00
talloc_free ( tmp_ctx ) ;
2011-04-11 15:14:52 +02:00
return err ;
2008-04-08 01:56:32 +02:00
}
2011-04-11 16:01:22 +02:00
static sbcErr smbconf_reg_delete_includes ( struct smbconf_ctx * ctx ,
2008-04-09 22:21:15 +02:00
const char * service )
{
2011-04-11 16:01:22 +02:00
WERROR werr ;
2011-04-08 17:20:35 +02:00
sbcErr err ;
2008-04-09 22:21:15 +02:00
struct registry_key * key = NULL ;
TALLOC_CTX * tmp_ctx = talloc_stackframe ( ) ;
2011-04-08 17:20:35 +02:00
err = smbconf_reg_open_service_key ( tmp_ctx , ctx , service ,
REG_KEY_ALL , & key ) ;
if ( ! SBC_ERROR_IS_OK ( err ) ) {
2008-04-09 22:21:15 +02:00
goto done ;
}
if ( ! smbconf_value_exists ( key , INCLUDES_VALNAME ) ) {
2011-04-11 16:01:22 +02:00
err = SBC_ERR_OK ;
2008-04-09 22:21:15 +02:00
goto done ;
}
werr = reg_deletevalue ( key , INCLUDES_VALNAME ) ;
2011-04-11 16:01:22 +02:00
if ( ! W_ERROR_IS_OK ( werr ) ) {
err = SBC_ERR_ACCESS_DENIED ;
goto done ;
}
2008-04-09 22:21:15 +02:00
2011-04-11 16:01:22 +02:00
err = SBC_ERR_OK ;
2008-04-09 22:21:15 +02:00
done :
2009-01-21 17:16:37 +01:00
talloc_free ( tmp_ctx ) ;
2011-04-11 16:01:22 +02:00
return err ;
2008-04-09 22:21:15 +02:00
}
2008-04-08 01:56:32 +02:00
2011-04-11 17:24:13 +02:00
static sbcErr smbconf_reg_transaction_start ( struct smbconf_ctx * ctx )
2009-02-24 10:52:30 +01:00
{
2011-04-11 17:24:13 +02:00
WERROR werr ;
werr = regdb_transaction_start ( ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
return SBC_ERR_IO_FAILURE ;
}
return SBC_ERR_OK ;
2009-02-24 10:52:30 +01:00
}
2011-04-11 17:24:13 +02:00
static sbcErr smbconf_reg_transaction_commit ( struct smbconf_ctx * ctx )
2009-02-24 10:52:30 +01:00
{
2011-04-11 17:24:13 +02:00
WERROR werr ;
werr = regdb_transaction_commit ( ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
return SBC_ERR_IO_FAILURE ;
}
return SBC_ERR_OK ;
2009-02-24 10:52:30 +01:00
}
2011-04-11 17:24:13 +02:00
static sbcErr smbconf_reg_transaction_cancel ( struct smbconf_ctx * ctx )
2009-02-24 10:52:30 +01:00
{
2011-04-11 17:24:13 +02:00
WERROR werr ;
werr = regdb_transaction_cancel ( ) ;
if ( ! W_ERROR_IS_OK ( werr ) ) {
return SBC_ERR_IO_FAILURE ;
}
return SBC_ERR_OK ;
2009-02-24 10:52:30 +01:00
}
2008-03-21 16:26:50 +01:00
struct smbconf_ops smbconf_ops_reg = {
. init = smbconf_reg_init ,
. shutdown = smbconf_reg_shutdown ,
2008-10-20 23:52:02 +02:00
. requires_messaging = smbconf_reg_requires_messaging ,
2008-10-23 11:16:50 +02:00
. is_writeable = smbconf_reg_is_writeable ,
2008-03-21 16:26:50 +01:00
. open_conf = smbconf_reg_open ,
. close_conf = smbconf_reg_close ,
. get_csn = smbconf_reg_get_csn ,
. drop = smbconf_reg_drop ,
. get_share_names = smbconf_reg_get_share_names ,
. share_exists = smbconf_reg_share_exists ,
. create_share = smbconf_reg_create_share ,
. get_share = smbconf_reg_get_share ,
. delete_share = smbconf_reg_delete_share ,
. set_parameter = smbconf_reg_set_parameter ,
. get_parameter = smbconf_reg_get_parameter ,
2008-04-08 01:56:32 +02:00
. delete_parameter = smbconf_reg_delete_parameter ,
. get_includes = smbconf_reg_get_includes ,
. set_includes = smbconf_reg_set_includes ,
2008-04-09 22:21:15 +02:00
. delete_includes = smbconf_reg_delete_includes ,
2009-02-24 10:52:30 +01:00
. transaction_start = smbconf_reg_transaction_start ,
. transaction_commit = smbconf_reg_transaction_commit ,
. transaction_cancel = smbconf_reg_transaction_cancel ,
2008-03-21 16:26:50 +01:00
} ;
2008-03-21 16:35:52 +01:00
/**
* initialize the smbconf registry backend
* the only function that is exported from this module
*/
2011-04-07 17:19:03 +02:00
sbcErr smbconf_init_reg ( TALLOC_CTX * mem_ctx , struct smbconf_ctx * * conf_ctx ,
2008-03-21 17:55:31 +01:00
const char * path )
2008-03-21 16:26:50 +01:00
{
2008-04-13 16:26:14 +02:00
return smbconf_init_internal ( mem_ctx , conf_ctx , path , & smbconf_ops_reg ) ;
2008-03-21 16:26:50 +01:00
}