2006-07-11 22:01:26 +04:00
/*
* Unix SMB / Netbios implementation .
* Utility for managing share permissions
*
* Copyright ( C ) Tim Potter 2000
* Copyright ( C ) Jeremy Allison 2000
* Copyright ( C ) Jelmer Vernooij 2003
* Copyright ( C ) Gerald ( Jerry ) Carter 2005.
*
* 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 23:25:36 +04:00
* the Free Software Foundation ; either version 3 of the License , or
2006-07-11 22:01:26 +04:00
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
2007-07-10 09:23:25 +04:00
* along with this program ; if not , see < http : //www.gnu.org/licenses/>.
2006-07-11 22:01:26 +04:00
*/
2015-04-24 19:51:28 +03:00
struct cli_state ;
2006-07-11 22:01:26 +04:00
# include "includes.h"
2010-08-05 12:49:53 +04:00
# include "popt_common.h"
2010-10-12 08:27:50 +04:00
# include "../libcli/security/security.h"
2011-03-22 18:50:02 +03:00
# include "passdb/machine_sid.h"
2015-04-24 19:51:28 +03:00
# include "util_sd.h"
2006-07-11 22:01:26 +04:00
2007-02-18 04:31:50 +03:00
static TALLOC_CTX * ctx ;
2006-07-11 22:01:26 +04:00
2008-12-25 04:03:22 +03:00
enum acl_mode { SMB_ACL_DELETE ,
SMB_ACL_MODIFY ,
SMB_ACL_ADD ,
SMB_ACL_SET ,
2008-12-29 04:22:28 +03:00
SMB_SD_DELETE ,
2011-09-15 21:27:07 +04:00
SMB_SD_SETSDDL ,
SMB_SD_VIEWSDDL ,
2013-06-26 17:21:39 +04:00
SMB_ACL_VIEW ,
SMB_ACL_VIEW_ALL } ;
2006-07-11 22:01:26 +04:00
/********************************************************************
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-05-18 12:29:34 +04:00
static struct security_descriptor * parse_acl_string ( TALLOC_CTX * mem_ctx , const char * szACL , size_t * sd_size )
2006-07-11 22:01:26 +04:00
{
2010-05-18 12:29:34 +04:00
struct security_descriptor * sd = NULL ;
2010-05-18 05:25:38 +04:00
struct security_ace * ace ;
2010-05-18 05:30:40 +04:00
struct security_acl * theacl ;
2006-07-11 22:01:26 +04:00
int num_ace ;
const char * pacl ;
int i ;
2008-12-25 04:03:22 +03:00
2006-07-11 22:01:26 +04:00
if ( ! szACL )
return NULL ;
pacl = szACL ;
num_ace = count_chars ( pacl , ' , ' ) + 1 ;
2008-12-25 04:03:22 +03:00
2011-06-07 05:58:39 +04:00
if ( ! ( ace = talloc_zero_array ( mem_ctx , struct security_ace , num_ace ) ) )
2006-07-11 22:01:26 +04:00
return NULL ;
2008-12-25 04:03:22 +03:00
2006-07-11 22:01:26 +04:00
for ( i = 0 ; i < num_ace ; i + + ) {
char * end_acl = strchr_m ( pacl , ' , ' ) ;
fstring acl_string ;
strncpy ( acl_string , pacl , MIN ( PTR_DIFF ( end_acl , pacl ) , sizeof ( fstring ) - 1 ) ) ;
acl_string [ MIN ( PTR_DIFF ( end_acl , pacl ) , sizeof ( fstring ) - 1 ) ] = ' \0 ' ;
2008-12-25 04:03:22 +03:00
2015-04-24 20:00:19 +03:00
if ( ! parse_ace ( NULL , & ace [ i ] , acl_string ) )
2006-07-11 22:01:26 +04:00
return NULL ;
pacl = end_acl ;
pacl + + ;
}
2008-12-25 04:03:22 +03:00
2009-02-24 02:44:34 +03:00
if ( ! ( theacl = make_sec_acl ( mem_ctx , NT4_ACL_REVISION , num_ace , ace ) ) )
2006-07-11 22:01:26 +04:00
return NULL ;
2008-12-25 04:03:22 +03:00
2010-05-18 14:52:18 +04:00
sd = make_sec_desc ( mem_ctx , SD_REVISION , SEC_DESC_SELF_RELATIVE ,
2009-02-24 02:44:34 +03:00
NULL , NULL , NULL , theacl , sd_size ) ;
2006-07-11 22:01:26 +04:00
return sd ;
}
2010-05-18 05:30:40 +04:00
/* add an ACE to a list of ACEs in a struct security_acl */
static bool add_ace ( TALLOC_CTX * mem_ctx , struct security_acl * * the_acl , struct security_ace * ace )
2007-02-18 04:31:50 +03:00
{
2010-05-18 05:30:40 +04:00
struct security_acl * new_ace ;
2010-05-18 05:25:38 +04:00
struct security_ace * aces ;
2007-02-18 04:31:50 +03:00
if ( ! * the_acl ) {
return ( ( ( * the_acl ) = make_sec_acl ( mem_ctx , 3 , 1 , ace ) ) ! = NULL ) ;
}
2010-05-18 05:25:38 +04:00
if ( ! ( aces = SMB_CALLOC_ARRAY ( struct security_ace , 1 + ( * the_acl ) - > num_aces ) ) ) {
2007-02-18 04:31:50 +03:00
return False ;
}
2010-05-18 05:25:38 +04:00
memcpy ( aces , ( * the_acl ) - > aces , ( * the_acl ) - > num_aces * sizeof ( struct
security_ace ) ) ;
memcpy ( aces + ( * the_acl ) - > num_aces , ace , sizeof ( struct security_ace ) ) ;
2007-02-18 04:31:50 +03:00
new_ace = make_sec_acl ( mem_ctx , ( * the_acl ) - > revision , 1 + ( * the_acl ) - > num_aces , aces ) ;
SAFE_FREE ( aces ) ;
( * the_acl ) = new_ace ;
return True ;
}
/* The MSDN is contradictory over the ordering of ACE entries in an ACL.
However NT4 gives a " The information may have been modified by a
computer running Windows NT 5.0 " if denied ACEs do not appear before
allowed ACEs . */
2010-05-18 05:25:38 +04:00
static int ace_compare ( struct security_ace * ace1 , struct security_ace * ace2 )
2007-02-18 04:31:50 +03:00
{
2014-05-28 19:44:08 +04:00
if ( security_ace_equal ( ace1 , ace2 ) )
2007-02-18 04:31:50 +03:00
return 0 ;
2008-12-25 04:03:22 +03:00
if ( ace1 - > type ! = ace2 - > type )
2007-02-18 04:31:50 +03:00
return ace2 - > type - ace1 - > type ;
2010-08-26 17:48:50 +04:00
if ( dom_sid_compare ( & ace1 - > trustee , & ace2 - > trustee ) )
return dom_sid_compare ( & ace1 - > trustee , & ace2 - > trustee ) ;
2007-02-18 04:31:50 +03:00
2008-12-25 04:03:22 +03:00
if ( ace1 - > flags ! = ace2 - > flags )
2007-02-18 04:31:50 +03:00
return ace1 - > flags - ace2 - > flags ;
2008-12-25 04:03:22 +03:00
if ( ace1 - > access_mask ! = ace2 - > access_mask )
2007-02-18 04:31:50 +03:00
return ace1 - > access_mask - ace2 - > access_mask ;
2008-12-25 04:03:22 +03:00
if ( ace1 - > size ! = ace2 - > size )
2007-02-18 04:31:50 +03:00
return ace1 - > size - ace2 - > size ;
2010-05-18 05:25:38 +04:00
return memcmp ( ace1 , ace2 , sizeof ( struct security_ace ) ) ;
2007-02-18 04:31:50 +03:00
}
2010-05-18 05:30:40 +04:00
static void sort_acl ( struct security_acl * the_acl )
2007-02-18 04:31:50 +03:00
{
2015-05-07 03:00:06 +03:00
uint32_t i ;
2007-02-18 04:31:50 +03:00
if ( ! the_acl ) return ;
2010-02-14 02:03:55 +03:00
TYPESAFE_QSORT ( the_acl - > aces , the_acl - > num_aces , ace_compare ) ;
2007-02-18 04:31:50 +03:00
for ( i = 1 ; i < the_acl - > num_aces ; ) {
2014-05-28 19:44:08 +04:00
if ( security_ace_equal ( & the_acl - > aces [ i - 1 ] ,
& the_acl - > aces [ i ] ) ) {
2007-02-18 04:31:50 +03:00
int j ;
for ( j = i ; j < the_acl - > num_aces - 1 ; j + + ) {
the_acl - > aces [ j ] = the_acl - > aces [ j + 1 ] ;
}
the_acl - > num_aces - - ;
} else {
i + + ;
}
}
}
static int change_share_sec ( TALLOC_CTX * mem_ctx , const char * sharename , char * the_acl , enum acl_mode mode )
{
2010-05-18 12:29:34 +04:00
struct security_descriptor * sd = NULL ;
struct security_descriptor * old = NULL ;
2007-02-18 04:31:50 +03:00
size_t sd_size = 0 ;
2015-05-07 03:00:06 +03:00
uint32_t i , j ;
2008-12-25 04:03:22 +03:00
2008-12-29 04:22:28 +03:00
if ( mode ! = SMB_ACL_SET & & mode ! = SMB_SD_DELETE ) {
2007-02-18 04:31:50 +03:00
if ( ! ( old = get_share_security ( mem_ctx , sharename , & sd_size ) ) ) {
2008-12-29 04:22:28 +03:00
fprintf ( stderr , " Unable to retrieve permissions for share "
" [%s] \n " , sharename ) ;
2007-02-18 04:31:50 +03:00
return - 1 ;
}
}
2008-12-29 04:22:28 +03:00
if ( ( mode ! = SMB_ACL_VIEW & & mode ! = SMB_SD_DELETE ) & &
! ( sd = parse_acl_string ( mem_ctx , the_acl , & sd_size ) ) ) {
2007-02-18 04:31:50 +03:00
fprintf ( stderr , " Failed to parse acl \n " ) ;
return - 1 ;
}
2008-12-25 04:03:22 +03:00
2007-02-18 04:31:50 +03:00
switch ( mode ) {
2013-06-26 17:21:39 +04:00
case SMB_ACL_VIEW_ALL :
/* should not happen */
return 0 ;
2007-02-18 04:31:50 +03:00
case SMB_ACL_VIEW :
2015-06-09 19:50:18 +03:00
sec_desc_print ( NULL , stdout , old , false ) ;
2007-02-18 04:31:50 +03:00
return 0 ;
case SMB_ACL_DELETE :
for ( i = 0 ; sd - > dacl & & i < sd - > dacl - > num_aces ; i + + ) {
2007-10-19 04:40:25 +04:00
bool found = False ;
2007-02-18 04:31:50 +03:00
for ( j = 0 ; old - > dacl & & j < old - > dacl - > num_aces ; j + + ) {
2014-05-28 19:44:08 +04:00
if ( security_ace_equal ( & sd - > dacl - > aces [ i ] ,
& old - > dacl - > aces [ j ] ) ) {
2015-05-07 03:00:06 +03:00
uint32_t k ;
2007-02-18 04:31:50 +03:00
for ( k = j ; k < old - > dacl - > num_aces - 1 ; k + + ) {
old - > dacl - > aces [ k ] = old - > dacl - > aces [ k + 1 ] ;
}
old - > dacl - > num_aces - - ;
found = True ;
break ;
}
}
if ( ! found ) {
2015-04-24 19:51:28 +03:00
printf ( " ACL for ACE: " ) ;
2015-06-09 19:50:18 +03:00
print_ace ( NULL , stdout , & sd - > dacl - > aces [ i ] , false ) ;
2015-04-24 19:51:28 +03:00
printf ( " not found \n " ) ;
2007-02-18 04:31:50 +03:00
}
}
break ;
case SMB_ACL_MODIFY :
for ( i = 0 ; sd - > dacl & & i < sd - > dacl - > num_aces ; i + + ) {
2007-10-19 04:40:25 +04:00
bool found = False ;
2007-02-18 04:31:50 +03:00
for ( j = 0 ; old - > dacl & & j < old - > dacl - > num_aces ; j + + ) {
2010-08-26 17:48:50 +04:00
if ( dom_sid_equal ( & sd - > dacl - > aces [ i ] . trustee ,
2007-02-18 04:31:50 +03:00
& old - > dacl - > aces [ j ] . trustee ) ) {
old - > dacl - > aces [ j ] = sd - > dacl - > aces [ i ] ;
found = True ;
}
}
if ( ! found ) {
2007-12-15 23:53:26 +03:00
printf ( " ACL for SID %s not found \n " ,
sid_string_tos ( & sd - > dacl - > aces [ i ] . trustee ) ) ;
2007-02-18 04:31:50 +03:00
}
}
if ( sd - > owner_sid ) {
old - > owner_sid = sd - > owner_sid ;
}
if ( sd - > group_sid ) {
old - > group_sid = sd - > group_sid ;
}
break ;
case SMB_ACL_ADD :
for ( i = 0 ; sd - > dacl & & i < sd - > dacl - > num_aces ; i + + ) {
add_ace ( mem_ctx , & old - > dacl , & sd - > dacl - > aces [ i ] ) ;
}
break ;
case SMB_ACL_SET :
old = sd ;
break ;
2008-12-29 04:22:28 +03:00
case SMB_SD_DELETE :
if ( ! delete_share_security ( sharename ) ) {
fprintf ( stderr , " Failed to delete security descriptor for "
" share [%s] \n " , sharename ) ;
return - 1 ;
}
return 0 ;
2011-09-15 21:27:07 +04:00
default :
fprintf ( stderr , " invalid command \n " ) ;
return - 1 ;
2007-02-18 04:31:50 +03:00
}
/* Denied ACE entries must come before allowed ones */
sort_acl ( old - > dacl ) ;
if ( ! set_share_security ( sharename , old ) ) {
fprintf ( stderr , " Failed to store acl for share [%s] \n " , sharename ) ;
return 2 ;
}
return 0 ;
}
2011-09-15 21:27:07 +04:00
static int set_sharesec_sddl ( const char * sharename , const char * sddl )
{
struct security_descriptor * sd ;
bool ret ;
sd = sddl_decode ( talloc_tos ( ) , sddl , get_global_sam_sid ( ) ) ;
if ( sd = = NULL ) {
fprintf ( stderr , " Failed to parse acl \n " ) ;
return - 1 ;
}
ret = set_share_security ( sharename , sd ) ;
TALLOC_FREE ( sd ) ;
if ( ! ret ) {
fprintf ( stderr , " Failed to store acl for share [%s] \n " ,
sharename ) ;
return - 1 ;
}
return 0 ;
}
static int view_sharesec_sddl ( const char * sharename )
{
struct security_descriptor * sd ;
size_t sd_size ;
char * acl ;
sd = get_share_security ( talloc_tos ( ) , sharename , & sd_size ) ;
if ( sd = = NULL ) {
fprintf ( stderr , " Unable to retrieve permissions for share "
" [%s] \n " , sharename ) ;
return - 1 ;
}
acl = sddl_encode ( talloc_tos ( ) , sd , get_global_sam_sid ( ) ) ;
TALLOC_FREE ( sd ) ;
if ( acl = = NULL ) {
fprintf ( stderr , " Unable to sddl-encode permissions for share "
" [%s] \n " , sharename ) ;
return - 1 ;
}
printf ( " %s \n " , acl ) ;
TALLOC_FREE ( acl ) ;
return 0 ;
}
2006-07-11 22:01:26 +04:00
/********************************************************************
main program
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2013-06-26 17:21:39 +04:00
enum {
OPT_VIEW_ALL = 1000 ,
} ;
2006-07-11 22:01:26 +04:00
int main ( int argc , const char * argv [ ] )
{
int opt ;
2007-02-18 04:31:50 +03:00
int retval = 0 ;
enum acl_mode mode = SMB_ACL_SET ;
2006-07-11 22:01:26 +04:00
static char * the_acl = NULL ;
fstring sharename ;
2007-10-19 04:40:25 +04:00
bool force_acl = False ;
2006-07-11 22:01:26 +04:00
int snum ;
poptContext pc ;
2007-10-19 04:40:25 +04:00
bool initialize_sid = False ;
2006-07-11 22:01:26 +04:00
struct poptOption long_options [ ] = {
POPT_AUTOHELP
2008-12-29 04:22:28 +03:00
{ " remove " , ' r ' , POPT_ARG_STRING , & the_acl , ' r ' , " Remove ACEs " , " ACL " } ,
{ " modify " , ' m ' , POPT_ARG_STRING , & the_acl , ' m ' , " Modify existing ACEs " , " ACL " } ,
{ " add " , ' a ' , POPT_ARG_STRING , & the_acl , ' a ' , " Add ACEs " , " ACL " } ,
{ " replace " , ' R ' , POPT_ARG_STRING , & the_acl , ' R ' , " Overwrite share permission ACL " , " ACLS " } ,
{ " delete " , ' D ' , POPT_ARG_NONE , NULL , ' D ' , " Delete the entire security descriptor " } ,
2011-09-15 21:27:07 +04:00
{ " setsddl " , ' S ' , POPT_ARG_STRING , the_acl , ' S ' ,
" Set the SD in sddl format " } ,
{ " viewsddl " , ' V ' , POPT_ARG_NONE , the_acl , ' V ' ,
" View the SD in sddl format " } ,
2006-07-11 22:01:26 +04:00
{ " view " , ' v ' , POPT_ARG_NONE , NULL , ' v ' , " View current share permissions " } ,
2013-06-26 17:21:39 +04:00
{ " view-all " , 0 , POPT_ARG_NONE , NULL , OPT_VIEW_ALL ,
" View all current share permissions " } ,
2006-07-11 22:01:26 +04:00
{ " machine-sid " , ' M ' , POPT_ARG_NONE , NULL , ' M ' , " Initialize the machine SID " } ,
{ " force " , ' F ' , POPT_ARG_NONE , NULL , ' F ' , " Force storing the ACL " , " ACLS " } ,
POPT_COMMON_SAMBA
{ NULL }
} ;
2007-09-04 09:39:06 +04:00
if ( ! ( ctx = talloc_stackframe ( ) ) ) {
2006-07-11 22:01:26 +04:00
fprintf ( stderr , " Failed to initialize talloc context! \n " ) ;
return - 1 ;
}
/* set default debug level to 1 regardless of what smb.conf sets */
2010-10-29 07:19:32 +04:00
setup_logging ( " sharesec " , DEBUG_STDERR ) ;
2010-10-29 14:10:31 +04:00
2015-03-21 22:00:06 +03:00
smb_init_locale ( ) ;
2010-10-29 14:10:31 +04:00
2010-10-29 08:06:36 +04:00
lp_set_cmdline ( " log level " , " 1 " ) ;
2006-07-11 22:01:26 +04:00
2007-02-18 04:31:50 +03:00
pc = poptGetContext ( " sharesec " , argc , argv , long_options , 0 ) ;
2008-12-25 04:03:22 +03:00
2006-07-11 22:01:26 +04:00
poptSetOtherOptionHelp ( pc , " sharename \n " ) ;
while ( ( opt = poptGetNextOpt ( pc ) ) ! = - 1 ) {
switch ( opt ) {
case ' r ' :
the_acl = smb_xstrdup ( poptGetOptArg ( pc ) ) ;
2007-02-18 04:31:50 +03:00
mode = SMB_ACL_DELETE ;
2006-07-11 22:01:26 +04:00
break ;
case ' m ' :
the_acl = smb_xstrdup ( poptGetOptArg ( pc ) ) ;
mode = SMB_ACL_MODIFY ;
break ;
case ' a ' :
the_acl = smb_xstrdup ( poptGetOptArg ( pc ) ) ;
mode = SMB_ACL_ADD ;
break ;
2008-12-29 04:22:28 +03:00
2006-07-11 22:01:26 +04:00
case ' R ' :
the_acl = smb_xstrdup ( poptGetOptArg ( pc ) ) ;
2007-02-18 04:31:50 +03:00
mode = SMB_ACL_SET ;
2006-07-11 22:01:26 +04:00
break ;
2008-12-29 04:22:28 +03:00
case ' D ' :
mode = SMB_SD_DELETE ;
break ;
2011-09-15 21:27:07 +04:00
case ' S ' :
mode = SMB_SD_SETSDDL ;
the_acl = smb_xstrdup ( poptGetOptArg ( pc ) ) ;
break ;
case ' V ' :
mode = SMB_SD_VIEWSDDL ;
break ;
2006-07-11 22:01:26 +04:00
case ' v ' :
mode = SMB_ACL_VIEW ;
break ;
case ' F ' :
force_acl = True ;
break ;
2008-12-25 04:03:22 +03:00
2006-07-11 22:01:26 +04:00
case ' M ' :
initialize_sid = True ;
break ;
2013-06-26 17:21:39 +04:00
case OPT_VIEW_ALL :
mode = SMB_ACL_VIEW_ALL ;
break ;
2006-07-11 22:01:26 +04:00
}
}
2008-12-25 04:03:22 +03:00
2007-02-18 04:31:50 +03:00
setlinebuf ( stdout ) ;
2015-04-21 16:24:42 +03:00
lp_load_with_registry_shares ( get_dyn_CONFIGFILE ( ) ) ;
2007-02-18 04:31:50 +03:00
2006-07-11 22:01:26 +04:00
/* check for initializing secrets.tdb first */
2008-12-25 04:03:22 +03:00
2006-07-11 22:01:26 +04:00
if ( initialize_sid ) {
2010-05-21 05:25:01 +04:00
struct dom_sid * sid = get_global_sam_sid ( ) ;
2008-12-25 04:03:22 +03:00
2006-07-11 22:01:26 +04:00
if ( ! sid ) {
fprintf ( stderr , " Failed to retrieve Machine SID! \n " ) ;
return 3 ;
}
2008-12-25 04:03:22 +03:00
2007-12-15 23:53:26 +03:00
printf ( " %s \n " , sid_string_tos ( sid ) ) ;
2006-07-11 22:01:26 +04:00
return 0 ;
}
if ( mode = = SMB_ACL_VIEW & & force_acl ) {
fprintf ( stderr , " Invalid combination of -F and -v \n " ) ;
return - 1 ;
}
2013-06-26 17:21:39 +04:00
if ( mode = = SMB_ACL_VIEW_ALL ) {
int i ;
for ( i = 0 ; i < lp_numservices ( ) ; i + + ) {
TALLOC_CTX * frame = talloc_stackframe ( ) ;
const char * service = lp_servicename ( frame , i ) ;
if ( service = = NULL ) {
continue ;
}
printf ( " [%s] \n " , service ) ;
change_share_sec ( frame , service , NULL , SMB_ACL_VIEW ) ;
printf ( " \n " ) ;
TALLOC_FREE ( frame ) ;
}
goto done ;
}
2006-07-11 22:01:26 +04:00
/* get the sharename */
2008-12-25 04:03:22 +03:00
if ( ! poptPeekArg ( pc ) ) {
poptPrintUsage ( pc , stderr , 0 ) ;
2006-07-11 22:01:26 +04:00
return - 1 ;
}
2008-12-25 04:03:22 +03:00
2006-07-11 22:01:26 +04:00
fstrcpy ( sharename , poptGetArg ( pc ) ) ;
2008-12-25 04:03:22 +03:00
2006-07-11 22:01:26 +04:00
snum = lp_servicenumber ( sharename ) ;
2008-12-25 04:03:22 +03:00
2006-07-11 22:01:26 +04:00
if ( snum = = - 1 & & ! force_acl ) {
fprintf ( stderr , " Invalid sharename: %s \n " , sharename ) ;
return - 1 ;
}
2008-12-25 04:03:22 +03:00
2011-09-15 21:27:07 +04:00
switch ( mode ) {
case SMB_SD_SETSDDL :
retval = set_sharesec_sddl ( sharename , the_acl ) ;
break ;
case SMB_SD_VIEWSDDL :
retval = view_sharesec_sddl ( sharename ) ;
break ;
default :
retval = change_share_sec ( ctx , sharename , the_acl , mode ) ;
break ;
}
2008-12-25 04:03:22 +03:00
2013-06-26 17:21:39 +04:00
done :
2006-07-11 22:01:26 +04:00
talloc_destroy ( ctx ) ;
2007-02-18 04:31:50 +03:00
return retval ;
2006-07-11 22:01:26 +04:00
}