2013-06-21 17:56:08 +02:00
/*
2008-02-28 11:23:20 -05:00
Unix SMB / Netbios implementation .
SMB client library implementation
Copyright ( C ) Andrew Tridgell 1998
Copyright ( C ) Richard Sharpe 2000 , 2002
Copyright ( C ) John Terpstra 2000
2013-06-21 17:56:08 +02:00
Copyright ( C ) Tom Jansen ( Ninja ISD ) 2002
2008-02-28 11:23:20 -05:00
Copyright ( C ) Derrell Lipman 2003 - 2008
Copyright ( C ) Jeremy Allison 2007 , 2008
2010-12-31 10:07:45 +01:00
2008-02-28 11:23:20 -05:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 3 of the License , or
( at your option ) any later version .
2010-12-31 10:07:45 +01:00
2008-02-28 11:23:20 -05: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 .
2010-12-31 10:07:45 +01:00
2008-02-28 11:23:20 -05:00
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"
2011-05-06 11:47:43 +02:00
# include "libsmb/libsmb.h"
2008-02-28 11:23:20 -05:00
# include "libsmbclient.h"
# include "libsmb_internal.h"
2010-08-05 02:25:37 +02:00
# include "secrets.h"
2012-05-19 18:39:44 +02:00
# include "../libcli/smb/smbXcli_base.h"
2021-01-13 16:11:17 +01:00
# include "auth/credentials/credentials.h"
# include "auth/gensec/gensec.h"
# include "lib/param/param.h"
2022-04-19 21:24:36 +02:00
# include "../lib/util/smb_threads.h"
# include "../lib/util/smb_threads_internal.h"
2008-02-28 11:23:20 -05:00
/*
2013-06-21 17:56:08 +02:00
* Is the logging working / configfile read ?
2008-02-28 11:23:20 -05:00
*/
2009-05-13 14:37:17 -04:00
static bool SMBC_initialized = false ;
static unsigned int initialized_ctx_count = 0 ;
static void * initialized_ctx_count_mutex = NULL ;
/*
2022-08-17 12:07:38 +02:00
* Do some module - and library - wide initializations
2009-05-13 14:37:17 -04:00
*/
static void
SMBC_module_init ( void * punused )
{
bool conf_loaded = False ;
char * home = NULL ;
TALLOC_CTX * frame = talloc_stackframe ( ) ;
2010-12-31 10:07:45 +01:00
2010-10-29 14:19:32 +11:00
setup_logging ( " libsmbclient " , DEBUG_STDOUT ) ;
2009-05-13 14:37:17 -04:00
/* Here we would open the smb.conf file if needed ... */
2010-12-31 10:07:45 +01:00
2009-05-13 14:37:17 -04:00
home = getenv ( " HOME " ) ;
if ( home ) {
char * conf = NULL ;
if ( asprintf ( & conf , " %s/.smb/smb.conf " , home ) > 0 ) {
2011-07-26 12:46:08 +02:00
if ( lp_load_client ( conf ) ) {
2009-05-13 14:37:17 -04:00
conf_loaded = True ;
} else {
DEBUG ( 5 , ( " Could not load config file: %s \n " ,
conf ) ) ;
}
SAFE_FREE ( conf ) ;
}
}
2010-12-31 10:07:45 +01:00
2009-05-13 14:37:17 -04:00
if ( ! conf_loaded ) {
/*
* Well , if that failed , try the get_dyn_CONFIGFILE
* Which points to the standard locn , and if that
* fails , silently ignore it and use the internal
* defaults . . .
*/
2010-12-31 10:07:45 +01:00
2011-07-26 12:46:08 +02:00
if ( ! lp_load_client ( get_dyn_CONFIGFILE ( ) ) ) {
2009-05-13 14:37:17 -04:00
DEBUG ( 5 , ( " Could not load config file: %s \n " ,
get_dyn_CONFIGFILE ( ) ) ) ;
} else if ( home ) {
char * conf ;
/*
* We loaded the global config file . Now lets
* load user - specific modifications to the
* global config .
*/
if ( asprintf ( & conf ,
" %s/.smb/smb.conf.append " ,
home ) > 0 ) {
2011-07-26 12:46:08 +02:00
if ( ! lp_load_client_no_reinit ( conf ) ) {
2009-05-13 14:37:17 -04:00
DEBUG ( 10 ,
( " Could not append config file: "
" %s \n " ,
conf ) ) ;
}
SAFE_FREE ( conf ) ;
}
}
}
2010-12-31 10:07:45 +01:00
2009-05-13 14:37:17 -04:00
load_interfaces ( ) ; /* Load the list of interfaces ... */
2010-12-31 10:07:45 +01:00
2009-05-13 14:37:17 -04:00
reopen_logs ( ) ; /* Get logging working ... */
2010-12-31 10:07:45 +01:00
2009-05-13 14:37:17 -04:00
/*
* Block SIGPIPE ( from lib / util_sock . c : write ( ) )
* It is not needed and should not stop execution
*/
BlockSignals ( True , SIGPIPE ) ;
2010-12-31 10:07:45 +01:00
2009-05-13 14:37:17 -04:00
/* Create the mutex we'll use to protect initialized_ctx_count */
if ( SMB_THREAD_CREATE_MUTEX ( " initialized_ctx_count_mutex " ,
initialized_ctx_count_mutex ) ! = 0 ) {
smb_panic ( " SMBC_module_init: "
" failed to create 'initialized_ctx_count' mutex " ) ;
}
TALLOC_FREE ( frame ) ;
}
2009-05-29 01:08:36 +02:00
static void
2009-05-13 14:37:17 -04:00
SMBC_module_terminate ( void )
{
2012-07-29 11:25:04 +10:00
TALLOC_CTX * frame = talloc_stackframe ( ) ;
2009-05-13 14:37:17 -04:00
secrets_shutdown ( ) ;
gfree_all ( ) ;
SMBC_initialized = false ;
2012-07-29 11:25:04 +10:00
TALLOC_FREE ( frame ) ;
2009-05-13 14:37:17 -04:00
}
2008-02-28 11:23:20 -05:00
/*
* Get a new empty handle to fill in with your own info
*/
SMBCCTX *
smbc_new_context ( void )
{
2008-03-01 20:44:21 -05:00
SMBCCTX * context ;
2012-07-29 11:25:04 +10:00
TALLOC_CTX * frame = talloc_stackframe ( ) ;
2010-12-31 10:07:45 +01:00
2009-05-13 14:37:17 -04:00
/* The first call to this function should initialize the module */
SMB_THREAD_ONCE ( & SMBC_initialized , SMBC_module_init , NULL ) ;
2008-03-01 20:44:21 -05:00
/*
* All newly added context fields should be placed in
* SMBC_internal_data , not directly in SMBCCTX .
*/
context = SMB_MALLOC_P ( SMBCCTX ) ;
if ( ! context ) {
2012-07-29 11:25:04 +10:00
TALLOC_FREE ( frame ) ;
2008-03-01 20:44:21 -05:00
errno = ENOMEM ;
return NULL ;
}
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
ZERO_STRUCTP ( context ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
context - > internal = SMB_MALLOC_P ( struct SMBC_internal_data ) ;
if ( ! context - > internal ) {
2012-07-29 11:25:04 +10:00
TALLOC_FREE ( frame ) ;
2008-03-01 20:44:21 -05:00
SAFE_FREE ( context ) ;
errno = ENOMEM ;
return NULL ;
}
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
/* Initialize the context and establish reasonable defaults */
ZERO_STRUCTP ( context - > internal ) ;
2010-12-31 10:07:45 +01:00
2008-03-03 18:13:33 -05:00
smbc_setDebug ( context , 0 ) ;
smbc_setTimeout ( context , 20000 ) ;
2013-04-16 14:53:09 -07:00
smbc_setPort ( context , 0 ) ;
2010-12-31 10:07:45 +01:00
2008-03-03 18:13:33 -05:00
smbc_setOptionFullTimeNames ( context , False ) ;
smbc_setOptionOpenShareMode ( context , SMBC_SHAREMODE_DENY_NONE ) ;
2020-06-10 12:40:13 +02:00
smbc_setOptionSmbEncryptionLevel ( context , SMBC_ENCRYPTLEVEL_DEFAULT ) ;
2010-01-24 19:24:10 +01:00
smbc_setOptionUseCCache ( context , True ) ;
2009-01-17 13:33:25 -05:00
smbc_setOptionCaseSensitive ( context , False ) ;
2008-03-03 18:13:33 -05:00
smbc_setOptionBrowseMaxLmbCount ( context , 3 ) ; /* # LMBs to query */
smbc_setOptionUrlEncodeReaddirEntries ( context , False ) ;
smbc_setOptionOneSharePerServer ( context , False ) ;
2016-11-30 10:25:53 -08:00
if ( getenv ( " LIBSMBCLIENT_NO_CCACHE " ) ! = NULL ) {
smbc_setOptionUseCCache ( context , false ) ;
2010-01-26 10:51:32 +01:00
}
2010-12-31 10:07:45 +01:00
2008-03-03 18:13:33 -05:00
smbc_setFunctionAuthData ( context , SMBC_get_auth_data ) ;
smbc_setFunctionCheckServer ( context , SMBC_check_server ) ;
smbc_setFunctionRemoveUnusedServer ( context , SMBC_remove_unused_server ) ;
2010-12-31 10:07:45 +01:00
2008-03-03 18:13:33 -05:00
smbc_setOptionUserData ( context , NULL ) ;
smbc_setFunctionAddCachedServer ( context , SMBC_add_cached_server ) ;
smbc_setFunctionGetCachedServer ( context , SMBC_get_cached_server ) ;
smbc_setFunctionRemoveCachedServer ( context , SMBC_remove_cached_server ) ;
smbc_setFunctionPurgeCachedServers ( context , SMBC_purge_cached_servers ) ;
2010-12-31 10:07:45 +01:00
2008-03-03 18:13:33 -05:00
smbc_setFunctionOpen ( context , SMBC_open_ctx ) ;
smbc_setFunctionCreat ( context , SMBC_creat_ctx ) ;
smbc_setFunctionRead ( context , SMBC_read_ctx ) ;
2015-05-27 23:13:15 +01:00
smbc_setFunctionSplice ( context , SMBC_splice_ctx ) ;
2008-03-03 18:13:33 -05:00
smbc_setFunctionWrite ( context , SMBC_write_ctx ) ;
smbc_setFunctionClose ( context , SMBC_close_ctx ) ;
smbc_setFunctionUnlink ( context , SMBC_unlink_ctx ) ;
smbc_setFunctionRename ( context , SMBC_rename_ctx ) ;
smbc_setFunctionLseek ( context , SMBC_lseek_ctx ) ;
smbc_setFunctionFtruncate ( context , SMBC_ftruncate_ctx ) ;
smbc_setFunctionStat ( context , SMBC_stat_ctx ) ;
2009-02-12 10:39:17 -05:00
smbc_setFunctionStatVFS ( context , SMBC_statvfs_ctx ) ;
smbc_setFunctionFstatVFS ( context , SMBC_fstatvfs_ctx ) ;
2008-03-03 18:13:33 -05:00
smbc_setFunctionFstat ( context , SMBC_fstat_ctx ) ;
smbc_setFunctionOpendir ( context , SMBC_opendir_ctx ) ;
smbc_setFunctionClosedir ( context , SMBC_closedir_ctx ) ;
smbc_setFunctionReaddir ( context , SMBC_readdir_ctx ) ;
2018-04-09 10:10:28 -07:00
smbc_setFunctionReaddirPlus ( context , SMBC_readdirplus_ctx ) ;
2019-10-18 10:29:48 -07:00
smbc_setFunctionReaddirPlus2 ( context , SMBC_readdirplus2_ctx ) ;
2008-03-03 18:13:33 -05:00
smbc_setFunctionGetdents ( context , SMBC_getdents_ctx ) ;
smbc_setFunctionMkdir ( context , SMBC_mkdir_ctx ) ;
smbc_setFunctionRmdir ( context , SMBC_rmdir_ctx ) ;
smbc_setFunctionTelldir ( context , SMBC_telldir_ctx ) ;
smbc_setFunctionLseekdir ( context , SMBC_lseekdir_ctx ) ;
smbc_setFunctionFstatdir ( context , SMBC_fstatdir_ctx ) ;
2015-06-26 13:36:43 +02:00
smbc_setFunctionNotify ( context , SMBC_notify_ctx ) ;
2008-03-03 18:13:33 -05:00
smbc_setFunctionChmod ( context , SMBC_chmod_ctx ) ;
smbc_setFunctionUtimes ( context , SMBC_utimes_ctx ) ;
smbc_setFunctionSetxattr ( context , SMBC_setxattr_ctx ) ;
smbc_setFunctionGetxattr ( context , SMBC_getxattr_ctx ) ;
smbc_setFunctionRemovexattr ( context , SMBC_removexattr_ctx ) ;
smbc_setFunctionListxattr ( context , SMBC_listxattr_ctx ) ;
2010-12-31 10:07:45 +01:00
2008-03-03 18:13:33 -05:00
smbc_setFunctionOpenPrintJob ( context , SMBC_open_print_job_ctx ) ;
smbc_setFunctionPrintFile ( context , SMBC_print_file_ctx ) ;
smbc_setFunctionListPrintJobs ( context , SMBC_list_print_jobs_ctx ) ;
smbc_setFunctionUnlinkPrintJob ( context , SMBC_unlink_print_job_ctx ) ;
2010-12-31 10:07:45 +01:00
2012-07-29 11:25:04 +10:00
TALLOC_FREE ( frame ) ;
2008-03-01 20:44:21 -05:00
return context ;
2008-02-28 11:23:20 -05:00
}
/*
* Free a context
*
* Returns 0 on success . Otherwise returns 1 , the SMBCCTX is _not_ freed
* and thus you ' ll be leaking memory if not handled properly .
*
*/
int
smbc_free_context ( SMBCCTX * context ,
int shutdown_ctx )
{
2012-07-29 11:25:04 +10:00
TALLOC_CTX * frame ;
2008-03-01 20:44:21 -05:00
if ( ! context ) {
errno = EBADF ;
return 1 ;
2008-02-28 11:23:20 -05:00
}
2010-12-31 10:07:45 +01:00
2012-07-29 11:25:04 +10:00
frame = talloc_stackframe ( ) ;
2008-03-01 20:44:21 -05:00
if ( shutdown_ctx ) {
SMBCFILE * f ;
DEBUG ( 1 , ( " Performing aggressive shutdown. \n " ) ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
f = context - > internal - > files ;
while ( f ) {
2016-04-13 17:57:31 +01:00
SMBCFILE * next = f - > next ;
2008-03-03 18:13:33 -05:00
smbc_getFunctionClose ( context ) ( context , f ) ;
2016-04-13 17:57:31 +01:00
f = next ;
2008-03-01 20:44:21 -05:00
}
context - > internal - > files = NULL ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
/* First try to remove the servers the nice way. */
2008-03-03 18:13:33 -05:00
if ( smbc_getFunctionPurgeCachedServers ( context ) ( context ) ) {
2008-03-01 20:44:21 -05:00
SMBCSRV * s ;
SMBCSRV * next ;
DEBUG ( 1 , ( " Could not purge all servers, "
" Nice way shutdown failed. \n " ) ) ;
s = context - > internal - > servers ;
while ( s ) {
2011-07-11 14:29:28 +02:00
DEBUG ( 1 , ( " Forced shutdown: %p (cli=%p) \n " ,
s , s - > cli ) ) ;
2008-03-01 20:44:21 -05:00
cli_shutdown ( s - > cli ) ;
2008-03-03 18:13:33 -05:00
smbc_getFunctionRemoveCachedServer ( context ) ( context ,
2008-03-01 20:44:21 -05:00
s ) ;
next = s - > next ;
DLIST_REMOVE ( context - > internal - > servers , s ) ;
SAFE_FREE ( s ) ;
s = next ;
}
context - > internal - > servers = NULL ;
}
2008-02-28 11:23:20 -05:00
}
2008-03-01 20:44:21 -05:00
else {
/* This is the polite way */
2008-03-03 18:13:33 -05:00
if ( smbc_getFunctionPurgeCachedServers ( context ) ( context ) ) {
2008-03-01 20:44:21 -05:00
DEBUG ( 1 , ( " Could not purge all servers, "
" free_context failed. \n " ) ) ;
errno = EBUSY ;
2012-07-29 11:25:04 +10:00
TALLOC_FREE ( frame ) ;
2008-03-01 20:44:21 -05:00
return 1 ;
}
if ( context - > internal - > servers ) {
DEBUG ( 1 , ( " Active servers in context, "
" free_context failed. \n " ) ) ;
errno = EBUSY ;
2012-07-29 11:25:04 +10:00
TALLOC_FREE ( frame ) ;
2008-03-01 20:44:21 -05:00
return 1 ;
}
if ( context - > internal - > files ) {
DEBUG ( 1 , ( " Active files in context, "
" free_context failed. \n " ) ) ;
errno = EBUSY ;
2012-07-29 11:25:04 +10:00
TALLOC_FREE ( frame ) ;
2008-03-01 20:44:21 -05:00
return 1 ;
}
2008-02-28 11:23:20 -05:00
}
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
/* Things we have to clean up */
2008-03-03 18:13:33 -05:00
smbc_setWorkgroup ( context , NULL ) ;
smbc_setNetbiosName ( context , NULL ) ;
smbc_setUser ( context , NULL ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
DEBUG ( 3 , ( " Context %p successfully freed \n " , context ) ) ;
2008-07-15 14:57:32 +02:00
2009-03-17 14:53:06 -07:00
/* Free any DFS auth context. */
2021-01-13 16:11:17 +01:00
TALLOC_FREE ( context - > internal - > creds ) ;
2009-03-17 14:53:06 -07:00
2008-08-12 13:35:15 -07:00
SAFE_FREE ( context - > internal ) ;
SAFE_FREE ( context ) ;
2008-07-15 14:57:32 +02:00
2009-05-13 14:37:17 -04:00
/* Protect access to the count of contexts in use */
if ( SMB_THREAD_LOCK ( initialized_ctx_count_mutex ) ! = 0 ) {
smb_panic ( " error locking 'initialized_ctx_count' " ) ;
}
2008-08-12 13:35:15 -07:00
if ( initialized_ctx_count ) {
initialized_ctx_count - - ;
}
2008-07-15 14:57:32 +02:00
2009-05-13 14:37:17 -04:00
if ( initialized_ctx_count = = 0 ) {
SMBC_module_terminate ( ) ;
}
/* Unlock the mutex */
if ( SMB_THREAD_UNLOCK ( initialized_ctx_count_mutex ) ! = 0 ) {
smb_panic ( " error unlocking 'initialized_ctx_count' " ) ;
2008-08-12 13:35:15 -07:00
}
2010-12-31 10:07:45 +01:00
2012-07-29 11:25:04 +10:00
TALLOC_FREE ( frame ) ;
2008-03-01 20:44:21 -05:00
return 0 ;
2008-02-28 11:23:20 -05:00
}
2008-03-01 20:44:21 -05:00
/**
* Deprecated interface . Do not use . Instead , use the various
* smbc_setOption * ( ) functions or smbc_setFunctionAuthDataWithContext ( ) .
2008-02-28 11:23:20 -05:00
*/
void
smbc_option_set ( SMBCCTX * context ,
char * option_name ,
. . . /* option_value */ )
{
2008-03-01 20:44:21 -05:00
va_list ap ;
union {
int i ;
bool b ;
smbc_get_auth_data_with_context_fn auth_fn ;
void * v ;
const char * s ;
} option_value ;
2010-12-31 10:07:45 +01:00
2012-07-29 11:25:04 +10:00
TALLOC_CTX * frame = talloc_stackframe ( ) ;
2008-03-01 20:44:21 -05:00
va_start ( ap , option_name ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
if ( strcmp ( option_name , " debug_to_stderr " ) = = 0 ) {
option_value . b = ( bool ) va_arg ( ap , int ) ;
smbc_setOptionDebugToStderr ( context , option_value . b ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " full_time_names " ) = = 0 ) {
option_value . b = ( bool ) va_arg ( ap , int ) ;
smbc_setOptionFullTimeNames ( context , option_value . b ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " open_share_mode " ) = = 0 ) {
option_value . i = va_arg ( ap , int ) ;
smbc_setOptionOpenShareMode ( context , option_value . i ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " auth_function " ) = = 0 ) {
option_value . auth_fn =
va_arg ( ap , smbc_get_auth_data_with_context_fn ) ;
smbc_setFunctionAuthDataWithContext ( context , option_value . auth_fn ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " user_data " ) = = 0 ) {
option_value . v = va_arg ( ap , void * ) ;
smbc_setOptionUserData ( context , option_value . v ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " smb_encrypt_level " ) = = 0 ) {
option_value . s = va_arg ( ap , const char * ) ;
if ( strcmp ( option_value . s , " none " ) = = 0 ) {
smbc_setOptionSmbEncryptionLevel ( context ,
SMBC_ENCRYPTLEVEL_NONE ) ;
} else if ( strcmp ( option_value . s , " request " ) = = 0 ) {
smbc_setOptionSmbEncryptionLevel ( context ,
SMBC_ENCRYPTLEVEL_REQUEST ) ;
} else if ( strcmp ( option_value . s , " require " ) = = 0 ) {
smbc_setOptionSmbEncryptionLevel ( context ,
SMBC_ENCRYPTLEVEL_REQUIRE ) ;
}
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " browse_max_lmb_count " ) = = 0 ) {
option_value . i = va_arg ( ap , int ) ;
smbc_setOptionBrowseMaxLmbCount ( context , option_value . i ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " urlencode_readdir_entries " ) = = 0 ) {
option_value . b = ( bool ) va_arg ( ap , int ) ;
smbc_setOptionUrlEncodeReaddirEntries ( context , option_value . b ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " one_share_per_server " ) = = 0 ) {
option_value . b = ( bool ) va_arg ( ap , int ) ;
smbc_setOptionOneSharePerServer ( context , option_value . b ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " use_kerberos " ) = = 0 ) {
option_value . b = ( bool ) va_arg ( ap , int ) ;
smbc_setOptionUseKerberos ( context , option_value . b ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " fallback_after_kerberos " ) = = 0 ) {
option_value . b = ( bool ) va_arg ( ap , int ) ;
smbc_setOptionFallbackAfterKerberos ( context , option_value . b ) ;
2010-12-31 10:07:45 +01:00
2010-01-24 19:24:10 +01:00
} else if ( strcmp ( option_name , " use_ccache " ) = = 0 ) {
option_value . b = ( bool ) va_arg ( ap , int ) ;
smbc_setOptionUseCCache ( context , option_value . b ) ;
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " no_auto_anonymous_login " ) = = 0 ) {
option_value . b = ( bool ) va_arg ( ap , int ) ;
smbc_setOptionNoAutoAnonymousLogin ( context , option_value . b ) ;
2008-02-29 13:34:35 -05:00
}
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
va_end ( ap ) ;
2012-07-29 11:25:04 +10:00
TALLOC_FREE ( frame ) ;
2008-02-28 11:23:20 -05:00
}
/*
2008-03-01 20:44:21 -05:00
* Deprecated interface . Do not use . Instead , use the various
* smbc_getOption * ( ) functions .
2008-02-28 11:23:20 -05:00
*/
void *
smbc_option_get ( SMBCCTX * context ,
char * option_name )
{
2008-03-01 20:44:21 -05:00
if ( strcmp ( option_name , " debug_stderr " ) = = 0 ) {
2008-02-28 11:23:20 -05:00
# if defined(__intptr_t_defined) || defined(HAVE_INTPTR_T)
2008-03-01 20:44:21 -05:00
return ( void * ) ( intptr_t ) smbc_getOptionDebugToStderr ( context ) ;
2008-02-28 11:23:20 -05:00
# else
2008-03-01 20:44:21 -05:00
return ( void * ) smbc_getOptionDebugToStderr ( context ) ;
2008-02-28 11:23:20 -05:00
# endif
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " full_time_names " ) = = 0 ) {
2008-02-28 11:23:20 -05:00
# if defined(__intptr_t_defined) || defined(HAVE_INTPTR_T)
2008-03-01 20:44:21 -05:00
return ( void * ) ( intptr_t ) smbc_getOptionFullTimeNames ( context ) ;
2008-02-28 11:23:20 -05:00
# else
2008-03-01 20:44:21 -05:00
return ( void * ) smbc_getOptionFullTimeNames ( context ) ;
2008-02-28 11:23:20 -05:00
# endif
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " open_share_mode " ) = = 0 ) {
2008-02-28 11:23:20 -05:00
# if defined(__intptr_t_defined) || defined(HAVE_INTPTR_T)
2008-03-01 20:44:21 -05:00
return ( void * ) ( intptr_t ) smbc_getOptionOpenShareMode ( context ) ;
2008-02-28 11:23:20 -05:00
# else
2008-03-01 20:44:21 -05:00
return ( void * ) smbc_getOptionOpenShareMode ( context ) ;
2008-02-28 11:23:20 -05:00
# endif
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " auth_function " ) = = 0 ) {
return ( void * ) smbc_getFunctionAuthDataWithContext ( context ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " user_data " ) = = 0 ) {
return smbc_getOptionUserData ( context ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " smb_encrypt_level " ) = = 0 ) {
switch ( smbc_getOptionSmbEncryptionLevel ( context ) )
{
2020-06-10 12:40:13 +02:00
case SMBC_ENCRYPTLEVEL_DEFAULT :
return discard_const_p ( void , " default " ) ;
2008-03-01 20:44:21 -05:00
case 0 :
2011-05-05 10:41:59 -07:00
return discard_const_p ( void , " none " ) ;
2008-03-01 20:44:21 -05:00
case 1 :
2011-05-05 10:41:59 -07:00
return discard_const_p ( void , " request " ) ;
2008-03-01 20:44:21 -05:00
case 2 :
2011-05-05 10:41:59 -07:00
return discard_const_p ( void , " require " ) ;
2008-03-01 20:44:21 -05:00
}
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " smb_encrypt_on " ) = = 0 ) {
SMBCSRV * s ;
unsigned int num_servers = 0 ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
for ( s = context - > internal - > servers ; s ; s = s - > next ) {
num_servers + + ;
2016-12-17 10:36:49 +01:00
if ( ! cli_state_is_encryption_on ( s - > cli ) ) {
2008-03-01 20:44:21 -05:00
return ( void * ) false ;
}
}
2008-02-28 11:23:20 -05:00
# if defined(__intptr_t_defined) || defined(HAVE_INTPTR_T)
2008-03-01 20:44:21 -05:00
return ( void * ) ( intptr_t ) ( bool ) ( num_servers > 0 ) ;
2008-02-28 11:23:20 -05:00
# else
2008-03-01 20:44:21 -05:00
return ( void * ) ( bool ) ( num_servers > 0 ) ;
2008-02-28 11:23:20 -05:00
# endif
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " browse_max_lmb_count " ) = = 0 ) {
2008-02-28 11:23:20 -05:00
# if defined(__intptr_t_defined) || defined(HAVE_INTPTR_T)
2008-03-01 20:44:21 -05:00
return ( void * ) ( intptr_t ) smbc_getOptionBrowseMaxLmbCount ( context ) ;
2008-02-28 11:23:20 -05:00
# else
2008-03-01 20:44:21 -05:00
return ( void * ) smbc_getOptionBrowseMaxLmbCount ( context ) ;
2008-02-28 11:23:20 -05:00
# endif
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " urlencode_readdir_entries " ) = = 0 ) {
2008-02-28 11:23:20 -05:00
# if defined(__intptr_t_defined) || defined(HAVE_INTPTR_T)
2008-03-01 20:44:21 -05:00
return ( void * ) ( intptr_t ) smbc_getOptionUrlEncodeReaddirEntries ( context ) ;
2008-02-28 11:23:20 -05:00
# else
2008-03-01 20:44:21 -05:00
return ( void * ) ( bool ) smbc_getOptionUrlEncodeReaddirEntries ( context ) ;
2008-02-28 11:23:20 -05:00
# endif
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " one_share_per_server " ) = = 0 ) {
2008-02-28 11:23:20 -05:00
# if defined(__intptr_t_defined) || defined(HAVE_INTPTR_T)
2008-03-01 20:44:21 -05:00
return ( void * ) ( intptr_t ) smbc_getOptionOneSharePerServer ( context ) ;
2008-02-28 11:23:20 -05:00
# else
2008-03-01 20:44:21 -05:00
return ( void * ) ( bool ) smbc_getOptionOneSharePerServer ( context ) ;
2008-02-28 11:23:20 -05:00
# endif
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " use_kerberos " ) = = 0 ) {
2008-02-28 11:23:20 -05:00
# if defined(__intptr_t_defined) || defined(HAVE_INTPTR_T)
2008-03-01 20:44:21 -05:00
return ( void * ) ( intptr_t ) smbc_getOptionUseKerberos ( context ) ;
2008-02-28 11:23:20 -05:00
# else
2008-03-01 20:44:21 -05:00
return ( void * ) ( bool ) smbc_getOptionUseKerberos ( context ) ;
2008-02-28 11:23:20 -05:00
# endif
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " fallback_after_kerberos " ) = = 0 ) {
2008-02-28 11:23:20 -05:00
# if defined(__intptr_t_defined) || defined(HAVE_INTPTR_T)
2008-03-01 20:44:21 -05:00
return ( void * ) ( intptr_t ) smbc_getOptionFallbackAfterKerberos ( context ) ;
2008-02-28 11:23:20 -05:00
# else
2008-03-01 20:44:21 -05:00
return ( void * ) ( bool ) smbc_getOptionFallbackAfterKerberos ( context ) ;
2008-02-28 11:23:20 -05:00
# endif
2010-12-31 10:07:45 +01:00
2010-01-24 19:24:10 +01:00
} else if ( strcmp ( option_name , " use_ccache " ) = = 0 ) {
# if defined(__intptr_t_defined) || defined(HAVE_INTPTR_T)
return ( void * ) ( intptr_t ) smbc_getOptionUseCCache ( context ) ;
# else
return ( void * ) ( bool ) smbc_getOptionUseCCache ( context ) ;
# endif
2008-03-01 20:44:21 -05:00
} else if ( strcmp ( option_name , " no_auto_anonymous_login " ) = = 0 ) {
# if defined(__intptr_t_defined) || defined(HAVE_INTPTR_T)
return ( void * ) ( intptr_t ) smbc_getOptionNoAutoAnonymousLogin ( context ) ;
# else
return ( void * ) ( bool ) smbc_getOptionNoAutoAnonymousLogin ( context ) ;
# endif
}
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
return NULL ;
2008-02-28 11:23:20 -05:00
}
/*
* Initialize the library , etc .
*
* We accept a struct containing handle information .
* valid values for info - > debug from 0 to 100 ,
* and insist that info - > fn must be non - null .
*/
SMBCCTX *
smbc_init_context ( SMBCCTX * context )
{
2008-03-01 20:44:21 -05:00
int pid ;
2014-04-01 17:03:34 +13:00
TALLOC_CTX * frame ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
if ( ! context ) {
errno = EBADF ;
return NULL ;
2008-02-28 11:23:20 -05:00
}
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
/* Do not initialise the same client twice */
if ( context - > internal - > initialized ) {
2008-03-01 21:19:52 -05:00
return NULL ;
2008-03-01 20:44:21 -05:00
}
2010-12-31 10:07:45 +01:00
2014-04-01 17:03:34 +13:00
frame = talloc_stackframe ( ) ;
2008-03-17 11:34:25 -04:00
if ( ( ! smbc_getFunctionAuthData ( context ) & &
! smbc_getFunctionAuthDataWithContext ( context ) ) | |
2008-03-03 18:13:33 -05:00
smbc_getDebug ( context ) < 0 | |
smbc_getDebug ( context ) > 100 ) {
2010-12-31 10:07:45 +01:00
2014-04-01 17:03:34 +13:00
TALLOC_FREE ( frame ) ;
2008-03-01 20:44:21 -05:00
errno = EINVAL ;
return NULL ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
}
2010-12-31 10:07:45 +01:00
2008-03-03 18:13:33 -05:00
if ( ! smbc_getUser ( context ) ) {
2008-02-28 11:23:20 -05:00
/*
2008-03-01 20:44:21 -05:00
* FIXME : Is this the best way to get the user info ?
2008-02-28 11:23:20 -05:00
*/
2010-01-15 17:22:35 -08:00
char * user = getenv ( " USER " ) ;
2008-03-01 20:44:21 -05:00
/* walk around as "guest" if no username can be found */
2008-03-03 18:13:33 -05:00
if ( ! user ) {
user = SMB_STRDUP ( " guest " ) ;
} else {
user = SMB_STRDUP ( user ) ;
}
if ( ! user ) {
2014-04-01 17:03:34 +13:00
TALLOC_FREE ( frame ) ;
2008-03-03 18:13:33 -05:00
errno = ENOMEM ;
return NULL ;
}
smbc_setUser ( context , user ) ;
2010-01-15 17:22:35 -08:00
SAFE_FREE ( user ) ;
if ( ! smbc_getUser ( context ) ) {
2014-04-01 17:03:34 +13:00
TALLOC_FREE ( frame ) ;
2010-01-15 17:22:35 -08:00
errno = ENOMEM ;
return NULL ;
}
2008-03-01 20:44:21 -05:00
}
2010-12-31 10:07:45 +01:00
2008-03-03 18:13:33 -05:00
if ( ! smbc_getNetbiosName ( context ) ) {
2008-03-01 20:44:21 -05:00
/*
* We try to get our netbios name from the config . If that
* fails we fall back on constructing our netbios name from
* our hostname etc
*/
2008-03-03 18:13:33 -05:00
char * netbios_name ;
2011-06-09 15:31:03 +10:00
if ( lp_netbios_name ( ) ) {
netbios_name = SMB_STRDUP ( lp_netbios_name ( ) ) ;
2008-03-03 18:13:33 -05:00
} else {
2008-03-01 20:44:21 -05:00
/*
* Hmmm , I want to get hostname as well , but I am too
* lazy for the moment
*/
2012-03-24 20:17:08 +01:00
pid = getpid ( ) ;
2008-03-03 18:13:33 -05:00
netbios_name = ( char * ) SMB_MALLOC ( 17 ) ;
if ( ! netbios_name ) {
2014-04-01 17:03:34 +13:00
TALLOC_FREE ( frame ) ;
2008-03-01 20:44:21 -05:00
errno = ENOMEM ;
return NULL ;
}
2008-03-03 18:13:33 -05:00
slprintf ( netbios_name , 16 ,
" smbc%s%d " , smbc_getUser ( context ) , pid ) ;
2008-03-01 20:44:21 -05:00
}
2008-03-03 18:13:33 -05:00
if ( ! netbios_name ) {
2014-04-01 17:03:34 +13:00
TALLOC_FREE ( frame ) ;
2008-03-03 18:13:33 -05:00
errno = ENOMEM ;
return NULL ;
}
2010-12-31 10:07:45 +01:00
2008-03-03 18:13:33 -05:00
smbc_setNetbiosName ( context , netbios_name ) ;
2010-01-15 17:22:35 -08:00
SAFE_FREE ( netbios_name ) ;
if ( ! smbc_getNetbiosName ( context ) ) {
2014-04-01 17:03:34 +13:00
TALLOC_FREE ( frame ) ;
2010-01-15 17:22:35 -08:00
errno = ENOMEM ;
return NULL ;
}
2008-03-01 20:44:21 -05:00
}
2010-12-31 10:07:45 +01:00
2008-03-03 18:13:33 -05:00
DEBUG ( 1 , ( " Using netbios name %s. \n " , smbc_getNetbiosName ( context ) ) ) ;
2010-12-31 10:07:45 +01:00
2008-03-03 18:13:33 -05:00
if ( ! smbc_getWorkgroup ( context ) ) {
2018-05-16 16:50:55 +02:00
const char * workgroup ;
2008-03-03 18:13:33 -05:00
2008-03-01 20:44:21 -05:00
if ( lp_workgroup ( ) ) {
2018-05-16 16:50:55 +02:00
workgroup = lp_workgroup ( ) ;
} else {
2008-03-01 20:44:21 -05:00
/* TODO: Think about a decent default workgroup */
2018-05-16 16:50:55 +02:00
workgroup = " samba " ;
2008-03-03 18:13:33 -05:00
}
smbc_setWorkgroup ( context , workgroup ) ;
2010-01-15 17:22:35 -08:00
if ( ! smbc_getWorkgroup ( context ) ) {
2014-04-01 17:03:34 +13:00
TALLOC_FREE ( frame ) ;
2010-01-15 17:22:35 -08:00
errno = ENOMEM ;
return NULL ;
}
2008-02-28 11:23:20 -05:00
}
2010-12-31 10:07:45 +01:00
2008-03-03 18:13:33 -05:00
DEBUG ( 1 , ( " Using workgroup %s. \n " , smbc_getWorkgroup ( context ) ) ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
/* shortest timeout is 1 second */
2008-03-03 18:13:33 -05:00
if ( smbc_getTimeout ( context ) > 0 & & smbc_getTimeout ( context ) < 1000 )
smbc_setTimeout ( context , 1000 ) ;
2010-12-31 10:07:45 +01:00
2008-03-01 20:44:21 -05:00
context - > internal - > initialized = True ;
2009-05-13 14:37:17 -04:00
/* Protect access to the count of contexts in use */
if ( SMB_THREAD_LOCK ( initialized_ctx_count_mutex ) ! = 0 ) {
smb_panic ( " error locking 'initialized_ctx_count' " ) ;
}
2008-08-12 13:35:15 -07:00
initialized_ctx_count + + ;
2009-05-13 14:37:17 -04:00
/* Unlock the mutex */
if ( SMB_THREAD_UNLOCK ( initialized_ctx_count_mutex ) ! = 0 ) {
smb_panic ( " error unlocking 'initialized_ctx_count' " ) ;
}
2010-12-31 10:07:45 +01:00
2014-04-01 17:03:34 +13:00
TALLOC_FREE ( frame ) ;
2008-03-01 20:44:21 -05:00
return context ;
2008-02-28 11:23:20 -05:00
}
2020-03-01 18:11:52 +01:00
/* Return the version of samba, and thus libsmbclient */
2008-02-28 11:23:20 -05:00
const char *
smbc_version ( void )
{
2009-01-15 22:27:52 +01:00
return samba_version_string ( ) ;
2008-02-28 11:23:20 -05:00
}
2008-07-08 20:44:39 -04:00
/*
* Set the credentials so DFS will work when following referrals .
2009-03-17 14:53:06 -07:00
* This function is broken and must be removed . No SMBCCTX arg . . .
* JRA .
2008-07-08 20:44:39 -04:00
*/
2009-03-17 14:53:06 -07:00
2008-07-08 20:44:39 -04:00
void
2009-02-23 13:50:11 -05:00
smbc_set_credentials ( const char * workgroup ,
2009-03-17 14:53:06 -07:00
const char * user ,
const char * password ,
smbc_bool use_kerberos ,
const char * signing_state )
2008-07-08 20:44:39 -04:00
{
2009-03-17 14:53:06 -07:00
d_printf ( " smbc_set_credentials is obsolete. Replace with smbc_set_credentials_with_fallback(). \n " ) ;
2008-07-08 20:44:39 -04:00
}
2009-02-20 12:00:46 +08:00
void smbc_set_credentials_with_fallback ( SMBCCTX * context ,
2009-02-20 16:41:41 -08:00
const char * workgroup ,
const char * user ,
const char * password )
2009-02-20 12:00:46 +08:00
{
2021-01-13 16:11:17 +01:00
struct loadparm_context * lp_ctx = NULL ;
struct cli_credentials * creds = NULL ;
enum credentials_use_kerberos kerberos_state =
CRED_USE_KERBEROS_DISABLED ;
2009-03-17 14:53:06 -07:00
2009-08-07 13:42:51 +08:00
if ( ! context ) {
return ;
}
if ( ! workgroup | | ! * workgroup ) {
workgroup = smbc_getWorkgroup ( context ) ;
2009-03-17 14:53:06 -07:00
}
2009-08-07 13:42:51 +08:00
if ( ! user ) {
user = smbc_getUser ( context ) ;
}
2009-02-20 09:51:36 -05:00
2009-08-07 13:42:51 +08:00
if ( ! password ) {
password = " " ;
2009-02-20 12:00:46 +08:00
}
2021-01-13 16:11:17 +01:00
creds = cli_credentials_init ( NULL ) ;
if ( creds = = NULL ) {
2009-03-17 14:53:06 -07:00
DEBUG ( 0 , ( " smbc_set_credentials_with_fallback: allocation fail \n " ) ) ;
return ;
}
2021-01-13 16:11:17 +01:00
lp_ctx = loadparm_init_s3 ( creds , loadparm_s3_helpers ( ) ) ;
if ( lp_ctx = = NULL ) {
TALLOC_FREE ( creds ) ;
return ;
2009-02-20 12:00:46 +08:00
}
2021-01-13 16:11:17 +01:00
cli_credentials_set_conf ( creds , lp_ctx ) ;
2009-02-20 12:00:46 +08:00
2021-01-13 16:11:17 +01:00
if ( smbc_getOptionUseKerberos ( context ) ) {
kerberos_state = CRED_USE_KERBEROS_REQUIRED ;
2009-02-20 12:00:46 +08:00
2021-01-13 16:11:17 +01:00
if ( smbc_getOptionFallbackAfterKerberos ( context ) ) {
kerberos_state = CRED_USE_KERBEROS_DESIRED ;
}
}
2009-02-20 12:00:46 +08:00
2021-01-13 16:11:17 +01:00
cli_credentials_set_username ( creds , user , CRED_SPECIFIED ) ;
cli_credentials_set_password ( creds , password , CRED_SPECIFIED ) ;
cli_credentials_set_domain ( creds , workgroup , CRED_SPECIFIED ) ;
cli_credentials_set_kerberos_state ( creds ,
kerberos_state ,
CRED_SPECIFIED ) ;
if ( smbc_getOptionUseCCache ( context ) ) {
uint32_t gensec_features ;
gensec_features = cli_credentials_get_gensec_features ( creds ) ;
gensec_features | = GENSEC_FEATURE_NTLM_CCACHE ;
cli_credentials_set_gensec_features ( creds ,
gensec_features ,
CRED_SPECIFIED ) ;
}
2009-03-17 14:53:06 -07:00
2021-01-13 16:11:17 +01:00
TALLOC_FREE ( context - > internal - > creds ) ;
context - > internal - > creds = creds ;
2009-02-20 12:00:46 +08:00
}