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
process model : standard ( 1 process per client connection )
2005-01-30 00:54:57 +00:00
Copyright ( C ) Andrew Tridgell 1992 - 2005
2003-08-13 01:53:07 +00:00
Copyright ( C ) James J Myers 2003 < myersjj @ samba . org >
2004-07-13 21:04:56 +00:00
Copyright ( C ) Stefan ( metze ) Metzmacher 2004
2003-08-13 01:53:07 +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-10 02:07:03 +00:00
the Free Software Foundation ; either version 3 of the License , or
2003-08-13 01:53:07 +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 02:07:03 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2003-08-13 01:53:07 +00:00
*/
# include "includes.h"
2005-02-03 11:56:03 +00:00
# include "lib/events/events.h"
2005-02-10 03:48:43 +00:00
# include "lib/tdb/include/tdb.h"
2006-03-09 17:48:41 +00:00
# include "lib/socket/socket.h"
2006-03-26 01:23:40 +00:00
# include "smbd/process_model.h"
2006-11-06 16:11:52 +00:00
# include "param/secrets.h"
2006-11-07 12:03:01 +00:00
# include "system/filesys.h"
2007-01-10 10:52:09 +00:00
# include "cluster/cluster.h"
2006-11-07 12:03:01 +00:00
2006-03-09 17:48:41 +00:00
# ifdef HAVE_SETPROCTITLE
2006-04-24 16:01:33 +00:00
# ifdef HAVE_SETPROCTITLE_H
2006-03-09 17:48:41 +00:00
# include <setproctitle.h>
2006-04-24 16:01:33 +00:00
# endif
2006-03-09 17:48:41 +00:00
# else
2006-04-26 12:54:52 +00:00
# define setproctitle none_setproctitle
static int none_setproctitle ( const char * fmt , . . . ) PRINTF_ATTRIBUTE ( 1 , 2 ) ;
static int none_setproctitle ( const char * fmt , . . . )
2006-04-26 12:15:01 +00:00
{
return 0 ;
}
2006-03-09 17:48:41 +00:00
# endif
2003-08-13 01:53:07 +00:00
/*
called when the process model is selected
*/
2005-01-30 00:54:57 +00:00
static void standard_model_init ( struct event_context * ev )
2003-08-13 01:53:07 +00:00
{
2004-04-07 07:17:11 +00:00
signal ( SIGCHLD , SIG_IGN ) ;
2005-01-14 01:32:56 +00:00
}
2003-08-13 01:53:07 +00:00
/*
2005-01-30 00:54:57 +00:00
called when a listening socket becomes readable .
2003-08-13 01:53:07 +00:00
*/
2005-01-30 00:54:57 +00:00
static void standard_accept_connection ( struct event_context * ev ,
struct socket_context * sock ,
void ( * new_conn ) ( struct event_context * , struct socket_context * ,
2007-01-10 10:52:09 +00:00
struct server_id , void * ) ,
2005-01-30 00:54:57 +00:00
void * private )
2003-08-13 01:53:07 +00:00
{
2004-09-20 12:31:07 +00:00
NTSTATUS status ;
2005-01-30 00:54:57 +00:00
struct socket_context * sock2 ;
2004-09-20 12:31:07 +00:00
pid_t pid ;
2005-01-30 00:54:57 +00:00
struct event_context * ev2 ;
2006-03-09 17:48:41 +00:00
struct socket_address * c , * s ;
2004-07-13 21:04:56 +00:00
/* accept an incoming connection. */
2005-01-30 00:54:57 +00:00
status = socket_accept ( sock , & sock2 ) ;
2004-09-20 12:31:07 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2004-09-28 12:30:42 +00:00
DEBUG ( 0 , ( " standard_accept_connection: accept: %s \n " ,
2004-09-20 12:31:07 +00:00
nt_errstr ( status ) ) ) ;
2005-10-12 11:04:01 +00:00
/* this looks strange, but is correct. We need to throttle things until
the system clears enough resources to handle this new socket */
sleep ( 1 ) ;
2003-08-13 01:53:07 +00:00
return ;
}
pid = fork ( ) ;
if ( pid ! = 0 ) {
/* parent or error code ... */
2005-01-30 00:54:57 +00:00
talloc_free ( sock2 ) ;
2003-08-13 01:53:07 +00:00
/* go back to the event loop */
return ;
}
2006-03-09 17:48:41 +00:00
pid = getpid ( ) ;
2005-01-30 00:54:57 +00:00
/* This is now the child code. We need a completely new event_context to work with */
ev2 = event_context_init ( NULL ) ;
2003-08-13 01:53:07 +00:00
2005-01-30 00:54:57 +00:00
/* the service has given us a private pointer that
encapsulates the context it needs for this new connection -
everything else will be freed */
talloc_steal ( ev2 , private ) ;
talloc_steal ( private , sock2 ) ;
/* this will free all the listening sockets and all state that
is not associated with this new connection */
talloc_free ( sock ) ;
talloc_free ( ev ) ;
2004-10-29 07:00:14 +00:00
/* we don't care if the dup fails, as its only a select()
speed optimisation */
2005-01-30 00:54:57 +00:00
socket_dup ( sock2 ) ;
2003-08-13 01:53:07 +00:00
/* tdb needs special fork handling */
2006-04-17 11:42:59 +00:00
if ( tdb_reopen_all ( 1 ) = = - 1 ) {
2004-07-13 21:04:56 +00:00
DEBUG ( 0 , ( " standard_accept_connection: tdb_reopen_all failed. \n " ) ) ;
2003-08-13 01:53:07 +00:00
}
2004-07-14 12:14:07 +00:00
/* Ensure that the forked children do not expose identical random streams */
set_need_random_reseed ( ) ;
2006-03-09 17:48:41 +00:00
/* setup the process title */
c = socket_get_peer_addr ( sock2 , ev2 ) ;
s = socket_get_my_addr ( sock2 , ev2 ) ;
if ( s & & c ) {
2006-04-26 12:15:01 +00:00
setproctitle ( " conn c[%s:%u] s[%s:%u] server_id[%d] " ,
c - > addr , c - > port , s - > addr , s - > port , pid ) ;
2006-03-09 17:48:41 +00:00
}
talloc_free ( c ) ;
talloc_free ( s ) ;
2005-01-30 00:54:57 +00:00
/* setup this new connection */
2007-01-10 10:52:09 +00:00
new_conn ( ev2 , sock2 , cluster_id ( pid ) , private ) ;
2004-07-13 21:04:56 +00:00
2005-01-30 00:54:57 +00:00
/* we can't return to the top level here, as that event context is gone,
so we now process events in the new event context until there are no
more to process */
event_loop_wait ( ev2 ) ;
2004-09-26 03:05:04 +00:00
2005-01-30 00:54:57 +00:00
talloc_free ( ev2 ) ;
exit ( 0 ) ;
2003-08-13 01:53:07 +00:00
}
2005-01-30 02:55:30 +00:00
/*
called to create a new server task
*/
static void standard_new_task ( struct event_context * ev ,
2007-01-10 10:52:09 +00:00
void ( * new_task ) ( struct event_context * , struct server_id , void * ) ,
2005-01-30 02:55:30 +00:00
void * private )
{
pid_t pid ;
struct event_context * ev2 ;
pid = fork ( ) ;
if ( pid ! = 0 ) {
/* parent or error code ... go back to the event loop */
return ;
}
2006-03-09 17:48:41 +00:00
pid = getpid ( ) ;
2005-01-30 02:55:30 +00:00
/* This is now the child code. We need a completely new event_context to work with */
ev2 = event_context_init ( NULL ) ;
/* the service has given us a private pointer that
encapsulates the context it needs for this new connection -
everything else will be freed */
talloc_steal ( ev2 , private ) ;
/* this will free all the listening sockets and all state that
is not associated with this new connection */
talloc_free ( ev ) ;
/* tdb needs special fork handling */
2006-04-17 11:42:59 +00:00
if ( tdb_reopen_all ( 1 ) = = - 1 ) {
2005-01-30 02:55:30 +00:00
DEBUG ( 0 , ( " standard_accept_connection: tdb_reopen_all failed. \n " ) ) ;
}
/* Ensure that the forked children do not expose identical random streams */
set_need_random_reseed ( ) ;
2006-04-26 12:15:01 +00:00
setproctitle ( " task server_id[%d] " , pid ) ;
2006-03-09 17:48:41 +00:00
2005-01-30 02:55:30 +00:00
/* setup this new connection */
2007-01-10 10:52:09 +00:00
new_task ( ev2 , cluster_id ( pid ) , private ) ;
2005-01-30 02:55:30 +00:00
/* we can't return to the top level here, as that event context is gone,
so we now process events in the new event context until there are no
more to process */
event_loop_wait ( ev2 ) ;
talloc_free ( ev2 ) ;
exit ( 0 ) ;
}
2004-07-13 21:04:56 +00:00
2005-01-30 02:55:30 +00:00
/* called when a task goes down */
2007-09-07 13:31:15 +00:00
_NORETURN_ static void standard_terminate ( struct event_context * ev , const char * reason )
2003-12-13 23:25:15 +00:00
{
2005-01-30 02:55:30 +00:00
DEBUG ( 2 , ( " standard_terminate: reason[%s] \n " , reason ) ) ;
2004-07-15 09:43:32 +00:00
2004-09-26 06:44:08 +00:00
/* this init_iconv() has the effect of freeing the iconv context memory,
which makes leak checking easier */
init_iconv ( ) ;
2004-10-25 02:57:20 +00:00
/* the secrets db should really hang off the connection structure */
secrets_shutdown ( ) ;
2005-01-30 00:54:57 +00:00
talloc_free ( ev ) ;
2005-01-14 01:32:56 +00:00
/* terminate this process */
exit ( 0 ) ;
2004-02-02 13:43:03 +00:00
}
2006-03-09 17:48:41 +00:00
/* called to set a title of a task or connection */
static void standard_set_title ( struct event_context * ev , const char * title )
{
if ( title ) {
2006-04-26 12:15:01 +00:00
setproctitle ( " %s " , title ) ;
2006-03-09 17:48:41 +00:00
} else {
2006-04-26 12:15:01 +00:00
setproctitle ( NULL ) ;
2006-03-09 17:48:41 +00:00
}
}
2005-01-30 00:54:57 +00:00
2005-01-14 01:32:56 +00:00
static const struct model_ops standard_ops = {
. name = " standard " ,
. model_init = standard_model_init ,
. accept_connection = standard_accept_connection ,
2005-01-30 02:55:30 +00:00
. new_task = standard_new_task ,
. terminate = standard_terminate ,
2006-03-09 17:48:41 +00:00
. set_title = standard_set_title ,
2005-01-14 01:32:56 +00:00
} ;
2003-08-13 01:53:07 +00:00
/*
2004-02-02 13:43:03 +00:00
initialise the standard process model , registering ourselves with the process model subsystem
2003-08-13 01:53:07 +00:00
*/
2004-02-02 13:43:03 +00:00
NTSTATUS process_model_standard_init ( void )
2003-08-13 01:53:07 +00:00
{
2005-01-30 00:54:57 +00:00
return register_process_model ( & standard_ops ) ;
2003-08-13 01:53:07 +00:00
}