2002-10-30 11:52:36 +00:00
/*
Unix SMB / CIFS implementation .
module loading system
2003-05-06 02:34:59 +00:00
Copyright ( C ) Jelmer Vernooij 2002 - 2003
Copyright ( C ) Stefan ( metze ) Metzmacher 2003
2002-10-30 11:52:36 +00:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
2007-07-09 19:25:36 +00:00
the Free Software Foundation ; either version 3 of the License , or
2002-10-30 11:52:36 +00: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 00:52:41 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2002-10-30 11:52:36 +00:00
*/
# include "includes.h"
# ifdef HAVE_DLOPEN
2003-09-04 02:02:26 +00:00
/* Load a dynamic module. Only log a level 0 error if we are not checking
for the existence of a module ( probling ) . */
2007-10-18 17:40:25 -07:00
static NTSTATUS do_smb_load_module ( const char * module_name , bool is_probe )
2002-10-30 11:52:36 +00:00
{
void * handle ;
init_module_function * init ;
2003-04-28 17:48:48 +00:00
NTSTATUS status ;
2003-01-17 21:23:14 +00:00
const char * error ;
2002-10-30 11:52:36 +00:00
/* Always try to use LAZY symbol resolving; if the plugin has
* backwards compatibility , there might be symbols in the
* plugin referencing to old ( removed ) functions
*/
2008-11-01 03:49:49 +01:00
handle = dlopen ( module_name , RTLD_LAZY ) ;
2002-10-30 11:52:36 +00:00
2004-11-30 01:01:43 +00:00
/* This call should reset any possible non-fatal errors that
occured since last call to dl * functions */
2008-11-01 03:49:49 +01:00
error = dlerror ( ) ;
2004-11-30 01:01:43 +00:00
2002-10-30 11:52:36 +00:00
if ( ! handle ) {
2003-09-15 00:19:25 +00:00
int level = is_probe ? 3 : 0 ;
2003-11-19 22:57:56 +00:00
DEBUG ( level , ( " Error loading module '%s': %s \n " , module_name , error ? error : " " ) ) ;
2003-04-28 17:48:48 +00:00
return NT_STATUS_UNSUCCESSFUL ;
2002-10-30 11:52:36 +00:00
}
2008-11-01 03:49:49 +01:00
init = ( init_module_function * ) dlsym ( handle , " init_samba_module " ) ;
2002-10-30 11:52:36 +00:00
2008-11-01 03:49:49 +01:00
/* we must check dlerror() to determine if it worked, because
dlsym ( ) can validly return NULL */
error = dlerror ( ) ;
2003-01-17 21:23:14 +00:00
if ( error ) {
2008-06-23 07:14:46 +02:00
DEBUG ( 0 , ( " Error trying to resolve symbol 'init_samba_module' "
2008-06-20 14:30:02 +02:00
" in %s: %s \n " , module_name , error ) ) ;
2008-11-01 03:49:49 +01:00
dlclose ( handle ) ;
2003-04-28 17:48:48 +00:00
return NT_STATUS_UNSUCCESSFUL ;
2002-10-30 11:52:36 +00:00
}
DEBUG ( 2 , ( " Module '%s' loaded \n " , module_name ) ) ;
2006-05-17 00:51:42 +00:00
status = init ( ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " Module '%s' initialization failed: %s \n " ,
module_name , get_friendly_nt_error_msg ( status ) ) ) ;
2008-11-01 03:49:49 +01:00
dlclose ( handle ) ;
2006-05-17 00:51:42 +00:00
}
2003-04-14 22:23:02 +00:00
return status ;
}
2003-09-04 02:02:26 +00:00
NTSTATUS smb_load_module ( const char * module_name )
{
return do_smb_load_module ( module_name , False ) ;
}
2007-09-14 22:03:41 +00:00
/* Load all modules in list and return number of
2003-04-14 22:23:02 +00:00
* modules that has been successfully loaded */
int smb_load_modules ( const char * * modules )
{
int i ;
int success = 0 ;
for ( i = 0 ; modules [ i ] ; i + + ) {
2003-04-28 17:48:48 +00:00
if ( NT_STATUS_IS_OK ( smb_load_module ( modules [ i ] ) ) ) {
2003-04-14 22:23:02 +00:00
success + + ;
}
}
DEBUG ( 2 , ( " %d modules successfully loaded \n " , success ) ) ;
return success ;
}
2003-04-28 17:48:48 +00:00
NTSTATUS smb_probe_module ( const char * subsystem , const char * module )
2003-04-14 22:23:02 +00:00
{
2007-09-14 22:03:41 +00:00
char * full_path = NULL ;
TALLOC_CTX * ctx = talloc_stackframe ( ) ;
NTSTATUS status ;
2003-04-14 22:23:02 +00:00
/* Check for absolute path */
2003-05-03 04:34:13 +00:00
2007-09-14 22:03:41 +00:00
/* if we make any 'samba multibyte string'
calls here , we break
2003-05-03 04:34:13 +00:00
for loading string modules */
2003-09-04 02:02:26 +00:00
DEBUG ( 5 , ( " Probing module '%s' \n " , module ) ) ;
2007-09-25 22:43:42 +00:00
if ( module [ 0 ] = = ' / ' ) {
status = do_smb_load_module ( module , True ) ;
TALLOC_FREE ( ctx ) ;
return status ;
}
2007-09-14 22:03:41 +00:00
full_path = talloc_asprintf ( ctx ,
2011-06-06 14:37:06 +10:00
" %s/%s.%s " ,
modules_path ( ctx , subsystem ) ,
module ,
shlib_ext ( ) ) ;
2007-09-14 22:03:41 +00:00
if ( ! full_path ) {
TALLOC_FREE ( ctx ) ;
return NT_STATUS_NO_MEMORY ;
}
DEBUG ( 5 , ( " Probing module '%s': Trying to load from %s \n " ,
module , full_path ) ) ;
status = do_smb_load_module ( full_path , True ) ;
TALLOC_FREE ( ctx ) ;
return status ;
2002-10-30 11:52:36 +00:00
}
# else /* HAVE_DLOPEN */
2003-04-28 17:48:48 +00:00
NTSTATUS smb_load_module ( const char * module_name )
2003-04-14 22:23:02 +00:00
{
2003-05-03 04:34:13 +00:00
DEBUG ( 0 , ( " This samba executable has not been built with plugin support \n " ) ) ;
2003-04-28 17:48:48 +00:00
return NT_STATUS_NOT_SUPPORTED ;
2003-04-14 22:23:02 +00:00
}
int smb_load_modules ( const char * * modules )
{
2003-05-03 04:34:13 +00:00
DEBUG ( 0 , ( " This samba executable has not been built with plugin support \n " ) ) ;
2003-04-28 17:48:48 +00:00
return - 1 ;
2003-04-14 22:23:02 +00:00
}
2003-04-28 17:48:48 +00:00
NTSTATUS smb_probe_module ( const char * subsystem , const char * module )
2002-10-30 11:52:36 +00:00
{
2003-05-03 04:34:13 +00:00
DEBUG ( 0 , ( " This samba executable has not been built with plugin support, not probing \n " ) ) ;
2003-04-28 17:48:48 +00:00
return NT_STATUS_NOT_SUPPORTED ;
2002-10-30 11:52:36 +00:00
}
# endif /* HAVE_DLOPEN */
2003-04-14 22:23:02 +00:00
void init_modules ( void )
{
/* FIXME: This can cause undefined symbol errors :
* smb_register_vfs ( ) isn ' t available in nmbd , for example */
if ( lp_preload_modules ( ) )
smb_load_modules ( lp_preload_modules ( ) ) ;
}