2003-08-13 01:53:07 +00:00
/*
Unix SMB / CIFS implementation .
2005-01-30 00:54:57 +00:00
2003-08-13 01:53:07 +00:00
Main SMB server routines
2005-01-30 00:54:57 +00:00
Copyright ( C ) Andrew Tridgell 1992 - 2005
2003-08-13 01:53:07 +00:00
Copyright ( C ) Martin Pool 2002
Copyright ( C ) Jelmer Vernooij 2002
Copyright ( C ) James J Myers 2003 < myersjj @ samba . org >
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 2 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 , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include "includes.h"
2005-02-03 11:56:03 +00:00
# include "lib/events/events.h"
2005-01-18 10:10:35 +00:00
# include "version.h"
2004-11-02 02:57:18 +00:00
# include "lib/cmdline/popt_common.h"
2005-01-30 00:54:57 +00:00
# include "system/dir.h"
2005-02-10 05:09:35 +00:00
# include "system/filesys.h"
2006-02-23 12:48:13 +00:00
# include "build.h"
2006-03-09 17:39:24 +00:00
# include "ldb/include/ldb.h"
2005-12-27 21:11:09 +00:00
# include "registry/registry.h"
2005-12-27 22:51:30 +00:00
# include "ntvfs/ntvfs.h"
2005-12-28 15:38:36 +00:00
# include "ntptr/ntptr.h"
2005-12-30 22:46:16 +00:00
# include "auth/gensec/gensec.h"
2006-03-07 11:07:23 +00:00
# include "smbd/process_model.h"
2006-03-07 13:22:00 +00:00
# include "smbd/service.h"
2006-11-06 16:11:52 +00:00
# include "param/secrets.h"
2006-11-03 01:49:19 +00:00
# include "smbd/pidfile.h"
2007-01-19 04:09:32 +00:00
# include "cluster/ctdb/ctdb_cluster.h"
2003-08-13 01:53:07 +00:00
2005-01-30 00:54:57 +00:00
/*
2005-07-10 08:06:28 +00:00
recursively delete a directory tree
2005-01-30 00:54:57 +00:00
*/
2005-07-10 08:06:28 +00:00
static void recursive_delete ( const char * path )
2003-08-13 01:53:07 +00:00
{
2005-01-30 00:54:57 +00:00
DIR * dir ;
struct dirent * de ;
2003-08-13 01:53:07 +00:00
2005-01-30 00:54:57 +00:00
dir = opendir ( path ) ;
if ( ! dir ) {
return ;
2003-08-13 01:53:07 +00:00
}
2005-01-30 00:54:57 +00:00
for ( de = readdir ( dir ) ; de ; de = readdir ( dir ) ) {
2005-07-10 08:06:28 +00:00
char * fname ;
struct stat st ;
2006-04-24 00:16:51 +00:00
if ( ISDOT ( de - > d_name ) | | ISDOTDOT ( de - > d_name ) ) {
2005-07-10 08:06:28 +00:00
continue ;
}
fname = talloc_asprintf ( path , " %s/%s " , path , de - > d_name ) ;
if ( stat ( fname , & st ) ! = 0 ) {
continue ;
}
if ( S_ISDIR ( st . st_mode ) ) {
recursive_delete ( fname ) ;
talloc_free ( fname ) ;
continue ;
}
if ( unlink ( fname ) ! = 0 ) {
DEBUG ( 0 , ( " Unabled to delete '%s' - %s \n " ,
fname , strerror ( errno ) ) ) ;
smb_panic ( " unable to cleanup tmp files " ) ;
2005-01-30 00:54:57 +00:00
}
2005-07-10 08:06:28 +00:00
talloc_free ( fname ) ;
2003-08-13 01:53:07 +00:00
}
2005-01-30 00:54:57 +00:00
closedir ( dir ) ;
2005-07-10 08:06:28 +00:00
}
/*
cleanup temporary files . This is the new alternative to
TDB_CLEAR_IF_FIRST . Unfortunately TDB_CLEAR_IF_FIRST is not
efficient on unix systems due to the lack of scaling of the byte
range locking system . So instead of putting the burden on tdb to
cleanup tmp files , this function deletes them .
*/
static void cleanup_tmp_files ( void )
{
char * path ;
TALLOC_CTX * mem_ctx = talloc_new ( NULL ) ;
path = smbd_tmp_path ( mem_ctx , NULL ) ;
2005-01-30 00:54:57 +00:00
2005-07-10 08:06:28 +00:00
recursive_delete ( path ) ;
2005-01-30 00:54:57 +00:00
talloc_free ( mem_ctx ) ;
}
2003-08-13 01:53:07 +00:00
2006-03-08 12:31:57 +00:00
static void sig_hup ( int sig )
{
debug_schedule_reopen_logs ( ) ;
}
2006-09-11 04:47:56 +00:00
static void sig_term ( int sig )
{
# if HAVE_GETPGRP
static int done_sigterm ;
if ( done_sigterm = = 0 & & getpgrp ( ) = = getpid ( ) ) {
DEBUG ( 0 , ( " SIGTERM: killing children \n " ) ) ;
done_sigterm = 1 ;
kill ( - getpgrp ( ) , SIGTERM ) ;
}
# endif
exit ( 0 ) ;
}
2005-01-30 00:54:57 +00:00
/*
setup signal masks
*/
static void setup_signals ( void )
{
2003-08-13 01:53:07 +00:00
/* we are never interested in SIGPIPE */
BlockSignals ( True , SIGPIPE ) ;
# if defined(SIGFPE)
/* we are never interested in SIGFPE */
BlockSignals ( True , SIGFPE ) ;
# endif
2005-12-15 16:53:20 +00:00
/* We are no longer interested in USR1 */
BlockSignals ( True , SIGUSR1 ) ;
2003-08-13 01:53:07 +00:00
# if defined(SIGUSR2)
/* We are no longer interested in USR2 */
BlockSignals ( True , SIGUSR2 ) ;
# endif
/* POSIX demands that signals are inherited. If the invoking process has
* these signals masked , we will have problems , as we won ' t recieve them . */
BlockSignals ( False , SIGHUP ) ;
BlockSignals ( False , SIGTERM ) ;
2005-01-30 00:54:57 +00:00
2006-03-08 12:31:57 +00:00
CatchSignal ( SIGHUP , sig_hup ) ;
2006-09-11 04:47:56 +00:00
CatchSignal ( SIGTERM , sig_term ) ;
2005-12-15 16:53:20 +00:00
}
2005-01-30 00:54:57 +00:00
2005-05-17 06:20:54 +00:00
/*
handle io on stdin
*/
static void server_stdin_handler ( struct event_context * event_ctx , struct fd_event * fde ,
uint16_t flags , void * private )
{
2005-12-15 18:08:25 +00:00
const char * binary_name = private ;
2005-05-17 06:20:54 +00:00
uint8_t c ;
if ( read ( 0 , & c , 1 ) = = 0 ) {
2005-12-15 18:08:25 +00:00
DEBUG ( 0 , ( " %s: EOF on stdin - terminating \n " , binary_name ) ) ;
2006-09-11 04:47:56 +00:00
# if HAVE_GETPGRP
if ( getpgrp ( ) = = getpid ( ) ) {
kill ( - getpgrp ( ) , SIGTERM ) ;
}
# endif
2005-05-17 06:20:54 +00:00
exit ( 0 ) ;
}
}
2005-06-12 00:17:23 +00:00
/*
die if the user selected maximum runtime is exceeded
*/
static void max_runtime_handler ( struct event_context * ev , struct timed_event * te ,
struct timeval t , void * private )
{
2005-12-15 18:08:25 +00:00
const char * binary_name = private ;
DEBUG ( 0 , ( " %s: maximum runtime exceeded - terminating \n " , binary_name ) ) ;
2005-06-12 00:17:23 +00:00
exit ( 0 ) ;
}
2005-01-30 00:54:57 +00:00
/*
main server .
*/
2005-12-15 18:08:25 +00:00
static int binary_smbd_main ( const char * binary_name , int argc , const char * argv [ ] )
2005-01-30 00:54:57 +00:00
{
BOOL interactive = False ;
int opt ;
poptContext pc ;
2006-03-09 19:55:04 +00:00
init_module_fn static_init [ ] = STATIC_service_MODULES ;
2005-12-26 16:46:55 +00:00
init_module_fn * shared_init ;
2005-01-30 00:54:57 +00:00
struct event_context * event_ctx ;
NTSTATUS status ;
const char * model = " standard " ;
2005-06-12 00:17:23 +00:00
int max_runtime = 0 ;
2006-04-03 14:02:53 +00:00
enum {
OPT_INTERACTIVE = 1000 ,
OPT_PROCESS_MODEL
} ;
2005-01-30 00:54:57 +00:00
struct poptOption long_options [ ] = {
POPT_AUTOHELP
2006-04-03 14:02:53 +00:00
{ " interactive " , ' i ' , POPT_ARG_NONE , NULL , OPT_INTERACTIVE ,
2005-01-30 00:54:57 +00:00
" Run interactive (not a daemon) " , NULL } ,
2006-04-03 14:02:53 +00:00
{ " model " , ' M ' , POPT_ARG_STRING , NULL , OPT_PROCESS_MODEL ,
2005-01-30 00:54:57 +00:00
" Select process model " , " MODEL " } ,
2006-04-03 14:02:53 +00:00
{ " maximum-runtime " , 0 , POPT_ARG_INT , & max_runtime , 0 ,
2005-12-15 18:08:25 +00:00
" set maximum runtime of the server process, till autotermination " , " seconds " } ,
2005-06-13 08:12:39 +00:00
POPT_COMMON_SAMBA
2005-01-30 00:54:57 +00:00
POPT_COMMON_VERSION
2006-09-06 12:28:01 +00:00
{ NULL }
2005-01-30 00:54:57 +00:00
} ;
2005-12-15 18:08:25 +00:00
pc = poptGetContext ( binary_name , argc , argv , long_options , 0 ) ;
2005-01-30 00:54:57 +00:00
2006-04-03 14:02:53 +00:00
while ( ( opt = poptGetNextOpt ( pc ) ) ! = - 1 ) {
switch ( opt ) {
case OPT_INTERACTIVE :
interactive = True ;
break ;
case OPT_PROCESS_MODEL :
model = poptGetOptArg ( pc ) ;
break ;
}
}
2005-01-30 00:54:57 +00:00
poptFreeContext ( pc ) ;
2005-12-16 10:21:05 +00:00
setup_logging ( binary_name , interactive ? DEBUG_STDOUT : DEBUG_FILE ) ;
2005-01-30 00:54:57 +00:00
setup_signals ( ) ;
2003-08-13 01:53:07 +00:00
/* we want total control over the permissions on created files,
so set our umask to 0 */
umask ( 0 ) ;
2005-12-15 18:08:25 +00:00
DEBUG ( 0 , ( " %s version %s started. \n " , binary_name , SAMBA_VERSION_STRING ) ) ;
2007-01-01 17:29:50 +00:00
DEBUGADD ( 0 , ( " Copyright Andrew Tridgell and the Samba Team 1992-2007 \n " ) ) ;
2003-08-13 01:53:07 +00:00
2004-07-13 21:04:56 +00:00
if ( sizeof ( uint16_t ) < 2 | | sizeof ( uint32_t ) < 4 | | sizeof ( uint64_t ) < 8 ) {
2003-08-13 01:53:07 +00:00
DEBUG ( 0 , ( " ERROR: Samba is not configured correctly for the word size on your machine \n " ) ) ;
exit ( 1 ) ;
}
2004-07-13 21:04:56 +00:00
2005-01-30 00:54:57 +00:00
if ( ! interactive ) {
2003-08-13 01:53:07 +00:00
DEBUG ( 3 , ( " Becoming a daemon. \n " ) ) ;
2005-01-30 00:54:57 +00:00
become_daemon ( True ) ;
2003-08-13 01:53:07 +00:00
}
2005-01-30 00:54:57 +00:00
cleanup_tmp_files ( ) ;
2005-02-10 06:36:30 +00:00
if ( ! directory_exist ( lp_lockdir ( ) ) ) {
2003-08-13 01:53:07 +00:00
mkdir ( lp_lockdir ( ) , 0755 ) ;
}
2005-12-15 18:08:25 +00:00
pidfile_create ( binary_name ) ;
2005-01-30 00:54:57 +00:00
/* Do *not* remove this, until you have removed
* passdb / secrets . c , and proved that Samba still builds . . . */
/* Setup the SECRETS subsystem */
if ( ! secrets_init ( ) ) {
exit ( 1 ) ;
2003-08-13 01:53:07 +00:00
}
2006-03-09 17:39:24 +00:00
ldb_global_init ( ) ; /* FIXME: */
2006-07-23 18:43:07 +00:00
share_init ( ) ;
2006-03-09 17:39:24 +00:00
gensec_init ( ) ; /* FIXME: */
2004-04-19 11:21:50 +00:00
2005-12-26 21:58:31 +00:00
registry_init ( ) ; /* FIXME: maybe run this in the initialization function
of the winreg RPC server instead ? */
ntptr_init ( ) ; /* FIXME: maybe run this in the initialization function
of the spoolss RPC server instead ? */
ntvfs_init ( ) ; /* FIXME: maybe run this in the initialization functions
of the SMB [ , 2 ] server instead ? */
process_model_init ( ) ;
2005-12-26 16:46:55 +00:00
shared_init = load_samba_modules ( NULL , " service " ) ;
run_init_functions ( static_init ) ;
run_init_functions ( shared_init ) ;
talloc_free ( shared_init ) ;
2005-01-30 00:54:57 +00:00
/* the event context is the top level structure in smbd. Everything else
should hang off that */
event_ctx = event_context_init ( NULL ) ;
2005-01-14 01:32:56 +00:00
2007-01-19 04:09:32 +00:00
/* initialise clustering if needed */
2007-04-16 06:23:42 +00:00
cluster_ctdb_init ( event_ctx , model ) ;
2007-01-19 04:09:32 +00:00
2005-06-07 08:33:17 +00:00
if ( interactive ) {
/* catch EOF on stdin */
2005-06-11 02:50:47 +00:00
# ifdef SIGTTIN
signal ( SIGTTIN , SIG_IGN ) ;
# endif
2005-06-07 08:33:17 +00:00
event_add_fd ( event_ctx , event_ctx , 0 , EVENT_FD_READ ,
2005-12-15 18:08:25 +00:00
server_stdin_handler ,
discard_const ( binary_name ) ) ;
2005-06-07 08:33:17 +00:00
}
2005-05-17 06:20:54 +00:00
2005-06-12 00:17:23 +00:00
if ( max_runtime ) {
event_add_timed ( event_ctx , event_ctx ,
timeval_current_ofs ( max_runtime , 0 ) ,
2005-12-15 18:08:25 +00:00
max_runtime_handler ,
discard_const ( binary_name ) ) ;
2005-06-12 00:17:23 +00:00
}
2005-12-15 18:08:25 +00:00
DEBUG ( 0 , ( " %s: using '%s' process model \n " , binary_name , model ) ) ;
2005-01-30 00:54:57 +00:00
status = server_service_startup ( event_ctx , model , lp_server_services ( ) ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " Starting Services failed - %s \n " , nt_errstr ( status ) ) ) ;
2004-07-13 21:04:56 +00:00
return 1 ;
}
2003-08-13 01:53:07 +00:00
2005-01-30 00:54:57 +00:00
/* wait for events - this is where smbd sits for most of its
life */
event_loop_wait ( event_ctx ) ;
2005-01-14 01:32:56 +00:00
2005-01-30 00:54:57 +00:00
/* as everything hangs off this event context, freeing it
should initiate a clean shutdown of all services */
talloc_free ( event_ctx ) ;
2005-01-14 01:32:56 +00:00
return 0 ;
2004-07-13 21:04:56 +00:00
}
int main ( int argc , const char * argv [ ] )
{
2005-12-15 18:08:25 +00:00
return binary_smbd_main ( " smbd " , argc , argv ) ;
2003-08-13 01:53:07 +00:00
}