2011-02-24 10:24:16 +01:00
/*
* Unix SMB / CIFS implementation .
*
* SMBD RPC service callbacks
*
* Copyright ( c ) 2011 Andreas Schneider < asn @ 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 3 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 , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
2011-05-02 13:21:53 +02:00
# include "ntdomain.h"
2011-07-05 12:04:00 +02:00
# include "messages.h"
# include "librpc/rpc/dcerpc_ep.h"
2011-02-24 10:24:16 +01:00
# include "../librpc/gen_ndr/srv_epmapper.h"
# include "rpc_server/rpc_server.h"
2011-07-05 12:04:00 +02:00
# include "rpc_server/rpc_sock_helper.h"
2011-03-14 14:50:09 +01:00
# include "rpc_server/epmapper/srv_epmapper.h"
2011-02-24 10:24:16 +01:00
# define DAEMON_NAME "epmd"
void start_epmd ( struct tevent_context * ev_ctx ,
struct messaging_context * msg_ctx ) ;
static void epmd_reopen_logs ( void )
{
2012-07-18 15:07:23 +09:30
char * lfile = lp_logfile ( talloc_tos ( ) ) ;
2011-02-24 10:24:16 +01:00
int rc ;
if ( lfile = = NULL | | lfile [ 0 ] = = ' \0 ' ) {
rc = asprintf ( & lfile , " %s/log.%s " , get_dyn_LOGFILEBASE ( ) , DAEMON_NAME ) ;
if ( rc > 0 ) {
lp_set_logfile ( lfile ) ;
SAFE_FREE ( lfile ) ;
}
} else {
if ( strstr ( lfile , DAEMON_NAME ) = = NULL ) {
2012-07-18 15:07:23 +09:30
rc = asprintf ( & lfile , " %s.%s " ,
lp_logfile ( talloc_tos ( ) ) , DAEMON_NAME ) ;
2011-02-24 10:24:16 +01:00
if ( rc > 0 ) {
lp_set_logfile ( lfile ) ;
SAFE_FREE ( lfile ) ;
}
}
}
reopen_logs ( ) ;
}
static void epmd_smb_conf_updated ( struct messaging_context * msg ,
void * private_data ,
uint32_t msg_type ,
struct server_id server_id ,
DATA_BLOB * data )
{
DEBUG ( 10 , ( " Got message saying smb.conf was updated. Reloading. \n " ) ) ;
change_to_root_user ( ) ;
epmd_reopen_logs ( ) ;
}
static void epmd_sig_term_handler ( struct tevent_context * ev ,
struct tevent_signal * se ,
int signum ,
int count ,
void * siginfo ,
void * private_data )
{
2011-03-14 14:50:09 +01:00
rpc_epmapper_shutdown ( ) ;
2011-02-24 10:24:16 +01:00
exit_server_cleanly ( " termination signal " ) ;
}
static void epmd_setup_sig_term_handler ( struct tevent_context * ev_ctx )
{
struct tevent_signal * se ;
se = tevent_add_signal ( ev_ctx ,
ev_ctx ,
SIGTERM , 0 ,
epmd_sig_term_handler ,
NULL ) ;
if ( se = = NULL ) {
exit_server ( " failed to setup SIGTERM handler " ) ;
}
}
static void epmd_sig_hup_handler ( struct tevent_context * ev ,
struct tevent_signal * se ,
int signum ,
int count ,
void * siginfo ,
void * private_data )
{
change_to_root_user ( ) ;
DEBUG ( 1 , ( " Reloading printers after SIGHUP \n " ) ) ;
epmd_reopen_logs ( ) ;
}
static void epmd_setup_sig_hup_handler ( struct tevent_context * ev_ctx ,
struct messaging_context * msg_ctx )
{
struct tevent_signal * se ;
se = tevent_add_signal ( ev_ctx ,
ev_ctx ,
SIGHUP , 0 ,
epmd_sig_hup_handler ,
msg_ctx ) ;
if ( se = = NULL ) {
exit_server ( " failed to setup SIGHUP handler " ) ;
}
}
2011-03-14 14:50:09 +01:00
static bool epmapper_shutdown_cb ( void * ptr ) {
srv_epmapper_cleanup ( ) ;
return true ;
}
2011-02-24 10:24:16 +01:00
void start_epmd ( struct tevent_context * ev_ctx ,
struct messaging_context * msg_ctx )
{
2011-03-14 14:50:09 +01:00
struct rpc_srv_callbacks epmapper_cb ;
2011-02-24 10:24:16 +01:00
NTSTATUS status ;
pid_t pid ;
bool ok ;
int rc ;
2011-03-14 14:50:09 +01:00
epmapper_cb . init = NULL ;
epmapper_cb . shutdown = epmapper_shutdown_cb ;
epmapper_cb . private_data = NULL ;
2011-02-24 10:24:16 +01:00
DEBUG ( 1 , ( " Forking Endpoint Mapper Daemon \n " ) ) ;
2012-03-24 20:17:08 +01:00
pid = fork ( ) ;
2011-02-24 10:24:16 +01:00
if ( pid = = - 1 ) {
DEBUG ( 0 , ( " Failed to fork Endpoint Mapper [%s], aborting ... \n " ,
strerror ( errno ) ) ) ;
exit ( 1 ) ;
}
if ( pid ) {
/* parent */
return ;
}
2015-09-23 11:14:05 -07:00
status = smbd_reinit_after_fork ( msg_ctx , ev_ctx , true , " epmd " ) ;
2011-02-24 10:24:16 +01:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " reinit_after_fork() failed \n " ) ) ;
smb_panic ( " reinit_after_fork() failed " ) ;
}
epmd_reopen_logs ( ) ;
epmd_setup_sig_term_handler ( ev_ctx ) ;
epmd_setup_sig_hup_handler ( ev_ctx , msg_ctx ) ;
messaging_register ( msg_ctx ,
ev_ctx ,
MSG_SMB_CONF_UPDATED ,
epmd_smb_conf_updated ) ;
2011-03-14 14:50:09 +01:00
status = rpc_epmapper_init ( & epmapper_cb ) ;
2011-02-24 10:24:16 +01:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2015-06-03 13:33:00 +02:00
DEBUG ( 0 , ( " Failed to register epmd rpc interface! (%s) \n " ,
2011-02-24 10:24:16 +01:00
nt_errstr ( status ) ) ) ;
exit ( 1 ) ;
}
2011-07-05 12:04:00 +02:00
status = rpc_setup_tcpip_sockets ( ev_ctx ,
msg_ctx ,
& ndr_table_epmapper ,
NULL ,
135 ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " Failed to open epmd tcpip sockets! \n " ) ) ;
exit ( 1 ) ;
}
2011-02-24 10:24:16 +01:00
ok = setup_dcerpc_ncalrpc_socket ( ev_ctx ,
msg_ctx ,
2011-03-14 12:29:49 +01:00
" EPMAPPER " ,
2011-03-17 11:14:12 +01:00
srv_epmapper_delete_endpoints ) ;
2011-02-24 10:24:16 +01:00
if ( ! ok ) {
DEBUG ( 0 , ( " Failed to open epmd ncalrpc pipe! \n " ) ) ;
exit ( 1 ) ;
}
2011-07-18 11:42:43 +02:00
ok = setup_named_pipe_socket ( " epmapper " , ev_ctx , msg_ctx ) ;
2011-02-24 10:24:16 +01:00
if ( ! ok ) {
DEBUG ( 0 , ( " Failed to open epmd named pipe! \n " ) ) ;
exit ( 1 ) ;
}
2013-03-15 10:02:53 +11:00
DEBUG ( 1 , ( " Endpoint Mapper Daemon Started (%u) \n " , ( unsigned int ) getpid ( ) ) ) ;
2011-02-24 10:24:16 +01:00
/* loop forever */
rc = tevent_loop_wait ( ev_ctx ) ;
/* should not be reached */
DEBUG ( 0 , ( " background_queue: tevent_loop_wait() exited with %d - %s \n " ,
rc , ( rc = = 0 ) ? " out of events " : strerror ( errno ) ) ) ;
exit ( 1 ) ;
}
/* vim: set ts=8 sw=8 noet cindent ft=c.doxygen: */