2000-09-13 11:07:17 +04:00
/*
2002-01-30 09:08:46 +03:00
Unix SMB / CIFS implementation .
2003-04-14 07:59:25 +04:00
Send messages to other Samba daemons
Copyright ( C ) Tim Potter 2003
2000-09-13 11:07:17 +04:00
Copyright ( C ) Andrew Tridgell 1994 - 1998
2003-04-14 07:59:25 +04:00
Copyright ( C ) Martin Pool 2001 - 2002
2002-07-15 14:35:28 +04:00
Copyright ( C ) Simo Sorce 2002
2000-09-13 11:07:17 +04: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 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"
2003-04-14 07:59:25 +04:00
/* Default timeout value when waiting for replies (in seconds) */
2002-01-08 00:32:22 +03:00
2003-04-14 07:59:25 +04:00
# define DEFAULT_TIMEOUT 10
2000-09-13 11:07:17 +04:00
2003-04-14 07:59:25 +04:00
static int timeout = DEFAULT_TIMEOUT ;
static int num_replies ; /* Used by message callback fns */
2000-10-12 04:29:01 +04:00
2003-04-14 07:59:25 +04:00
/* Send a message to a destination pid. Zero means broadcast smbd. */
2000-10-12 04:29:01 +04:00
2003-06-28 00:55:48 +04:00
static BOOL send_message ( pid_t pid , int msg_type , const void * buf , int len ,
2003-04-14 07:59:25 +04:00
BOOL duplicates )
{
TDB_CONTEXT * tdb ;
BOOL ret ;
int n_sent = 0 ;
if ( ! message_init ( ) )
return False ;
if ( pid ! = 0 )
return message_send_pid ( pid , msg_type , buf , len , duplicates ) ;
tdb = tdb_open_log ( lock_path ( " connections.tdb " ) , 0 ,
TDB_DEFAULT , O_RDWR , 0 ) ;
if ( ! tdb ) {
fprintf ( stderr , " Failed to open connections database "
" : %s \n " , strerror ( errno ) ) ;
return False ;
}
ret = message_send_all ( tdb , msg_type , buf , len , duplicates ,
& n_sent ) ;
DEBUG ( 10 , ( " smbcontrol/send_message: broadcast message to "
" %d processes \n " , n_sent ) ) ;
tdb_close ( tdb ) ;
return ret ;
}
2002-11-26 03:46:31 +03:00
2003-04-14 07:59:25 +04:00
/* Wait for one or more reply messages */
static void wait_replies ( BOOL multiple_replies )
{
time_t start_time = time ( NULL ) ;
2002-11-26 03:46:31 +03:00
2003-04-14 07:59:25 +04:00
/* Wait around a bit. This is pretty disgusting - we have to
busy - wait here as there is no nicer way to do it . */
2002-11-26 03:46:31 +03:00
2003-04-14 07:59:25 +04:00
do {
message_dispatch ( ) ;
if ( num_replies > 0 & & ! multiple_replies )
break ;
sleep ( 1 ) ;
} while ( timeout - ( time ( NULL ) - start_time ) > 0 ) ;
}
/* Message handler callback that displays a string on stdout */
static void print_string_cb ( int msg_type , pid_t pid , void * buf , size_t len )
2000-09-13 11:07:17 +04:00
{
2003-04-14 07:59:25 +04:00
printf ( " %.*s " , ( int ) len , ( const char * ) buf ) ;
num_replies + + ;
}
/* Send no message. Useful for testing. */
2003-06-28 00:55:48 +04:00
static BOOL do_noop ( const pid_t pid , const int argc , const char * * argv )
2003-04-14 07:59:25 +04:00
{
if ( argc ! = 1 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> noop \n " ) ;
return False ;
2000-09-30 00:08:00 +04:00
}
2003-04-14 07:59:25 +04:00
/* Move along, nothing to see here */
return True ;
2000-09-13 11:07:17 +04:00
}
2003-04-14 07:59:25 +04:00
/* Send a debug string */
2003-06-28 00:55:48 +04:00
static BOOL do_debug ( const pid_t pid , const int argc , const char * * argv )
2001-12-21 07:28:03 +03:00
{
2003-04-14 07:59:25 +04:00
if ( argc ! = 2 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> debug "
" <debug-string> \n " ) ;
return False ;
}
2001-12-21 07:28:03 +03:00
2003-04-14 07:59:25 +04:00
return send_message (
pid , MSG_DEBUG , argv [ 1 ] , strlen ( argv [ 1 ] ) + 1 , False ) ;
}
/* Force a browser election */
2003-06-28 00:55:48 +04:00
static BOOL do_election ( const pid_t pid , const int argc , const char * * argv )
2003-04-14 07:59:25 +04:00
{
if ( argc ! = 1 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> force-election \n " ) ;
return False ;
2001-12-21 07:28:03 +03:00
}
2003-04-14 07:59:25 +04:00
return send_message (
pid , MSG_FORCE_ELECTION , NULL , 0 , False ) ;
2001-12-21 07:28:03 +03:00
}
2000-10-07 03:01:47 +04:00
2003-04-14 07:59:25 +04:00
/* Ping a samba daemon process */
2000-09-13 11:07:17 +04:00
2003-04-14 07:59:25 +04:00
static void pong_cb ( int msg_type , pid_t pid , void * buf , size_t len )
2000-09-13 11:07:17 +04:00
{
2003-04-14 07:59:25 +04:00
printf ( " PONG from pid %u \n " , ( unsigned int ) pid ) ;
num_replies + + ;
2000-10-07 03:01:47 +04:00
}
2003-06-28 00:55:48 +04:00
static BOOL do_ping ( const pid_t pid , const int argc , const char * * argv )
2002-12-20 23:21:31 +03:00
{
2003-04-14 07:59:25 +04:00
if ( argc ! = 1 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> ping \n " ) ;
return False ;
}
/* Send a message and register our interest in a reply */
if ( ! send_message ( pid , MSG_PING , NULL , 0 , False ) )
return False ;
message_register ( MSG_PONG , pong_cb ) ;
wait_replies ( pid = = 0 ) ;
/* No replies were received within the timeout period */
if ( num_replies = = 0 )
printf ( " No replies received \n " ) ;
message_deregister ( MSG_PONG ) ;
return num_replies ;
2002-12-20 23:21:31 +03:00
}
2003-04-14 07:59:25 +04:00
/* Set profiling options */
2003-06-28 00:55:48 +04:00
static BOOL do_profile ( const pid_t pid , const int argc , const char * * argv )
2000-10-07 03:01:47 +04:00
{
2003-04-14 07:59:25 +04:00
int v ;
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
if ( argc ! = 2 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> profile "
" <off|count|on|flush> \n " ) ;
return False ;
}
if ( strcmp ( argv [ 1 ] , " off " ) = = 0 ) {
v = 0 ;
} else if ( strcmp ( argv [ 1 ] , " count " ) = = 0 ) {
v = 1 ;
} else if ( strcmp ( argv [ 1 ] , " on " ) = = 0 ) {
v = 2 ;
} else if ( strcmp ( argv [ 1 ] , " flush " ) = = 0 ) {
v = 3 ;
} else {
fprintf ( stderr , " Unknown profile command '%s' \n " , argv [ 1 ] ) ;
return False ;
}
return send_message ( pid , MSG_PROFILE , & v , sizeof ( int ) , False ) ;
2000-09-13 11:07:17 +04:00
}
2003-04-14 07:59:25 +04:00
/* Return the profiling level */
static void profilelevel_cb ( int msg_type , pid_t pid , void * buf , size_t len )
2000-11-11 03:33:33 +03:00
{
2003-04-14 07:59:25 +04:00
int level ;
const char * s ;
2000-11-11 03:33:33 +03:00
2003-04-14 07:59:25 +04:00
num_replies + + ;
if ( len ! = sizeof ( int ) ) {
fprintf ( stderr , " invalid message length %d returned \n " , len ) ;
return ;
}
memcpy ( & level , buf , sizeof ( int ) ) ;
switch ( level ) {
case 0 :
s = " not enabled " ;
break ;
case 1 :
2000-11-11 03:33:33 +03:00
s = " off " ;
break ;
2003-04-14 07:59:25 +04:00
case 3 :
2000-11-11 03:33:33 +03:00
s = " count only " ;
break ;
2003-04-14 07:59:25 +04:00
case 7 :
2000-11-11 03:33:33 +03:00
s = " count and time " ;
break ;
2003-04-14 07:59:25 +04:00
default :
s = " BOGUS " ;
break ;
2000-11-11 03:33:33 +03:00
}
2003-04-14 07:59:25 +04:00
printf ( " Profiling %s on pid %u \n " , s , ( unsigned int ) pid ) ;
2000-11-11 03:33:33 +03:00
}
2003-04-14 07:59:25 +04:00
static void profilelevel_rqst ( int msg_type , pid_t pid , void * buf , size_t len )
2001-12-21 07:28:03 +03:00
{
2003-04-14 07:59:25 +04:00
int v = 0 ;
2001-12-21 07:28:03 +03:00
2003-04-14 07:59:25 +04:00
/* Send back a dummy reply */
send_message ( pid , MSG_PROFILELEVEL , & v , sizeof ( int ) , False ) ;
}
2001-12-21 07:28:03 +03:00
2003-06-28 00:55:48 +04:00
static BOOL do_profilelevel ( const pid_t pid , const int argc , const char * * argv )
2000-09-13 11:07:17 +04:00
{
2003-04-14 07:59:25 +04:00
if ( argc ! = 1 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> profilelevel \n " ) ;
return False ;
}
2001-05-15 05:28:34 +04:00
2003-04-14 07:59:25 +04:00
/* Send a message and register our interest in a reply */
if ( ! send_message ( pid , MSG_REQ_PROFILELEVEL , NULL , 0 , False ) )
return False ;
message_register ( MSG_PROFILELEVEL , profilelevel_cb ) ;
message_register ( MSG_REQ_PROFILELEVEL , profilelevel_rqst ) ;
wait_replies ( pid = = 0 ) ;
2000-09-13 11:07:17 +04:00
2003-04-14 07:59:25 +04:00
/* No replies were received within the timeout period */
if ( num_replies = = 0 )
printf ( " No replies received \n " ) ;
message_deregister ( MSG_PROFILE ) ;
return num_replies ;
2000-09-13 11:07:17 +04:00
}
2003-04-14 07:59:25 +04:00
/* Display debug level settings */
2003-06-28 00:55:48 +04:00
static BOOL do_debuglevel ( const pid_t pid , const int argc , const char * * argv )
2000-09-13 11:07:17 +04:00
{
2003-04-14 07:59:25 +04:00
if ( argc ! = 1 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> debuglevel \n " ) ;
return False ;
2000-09-13 11:07:17 +04:00
}
2003-04-14 07:59:25 +04:00
/* Send a message and register our interest in a reply */
2000-09-13 11:07:17 +04:00
2003-04-14 07:59:25 +04:00
if ( ! send_message ( pid , MSG_REQ_DEBUGLEVEL , NULL , 0 , False ) )
return False ;
2002-01-09 11:27:15 +03:00
2003-04-14 07:59:25 +04:00
message_register ( MSG_DEBUGLEVEL , print_string_cb ) ;
2002-08-17 18:45:04 +04:00
2003-04-14 07:59:25 +04:00
wait_replies ( pid = = 0 ) ;
2002-08-17 18:45:04 +04:00
2003-04-14 07:59:25 +04:00
/* No replies were received within the timeout period */
2002-08-17 18:45:04 +04:00
2003-04-14 07:59:25 +04:00
if ( num_replies = = 0 )
printf ( " No replies received \n " ) ;
2002-08-17 18:45:04 +04:00
2003-04-14 07:59:25 +04:00
message_deregister ( MSG_DEBUGLEVEL ) ;
2002-08-17 18:45:04 +04:00
2003-04-14 07:59:25 +04:00
return num_replies ;
2002-08-17 18:45:04 +04:00
}
2002-01-09 11:27:15 +03:00
2003-04-14 07:59:25 +04:00
/* Send a print notify message */
2003-06-28 00:55:48 +04:00
static BOOL do_printnotify ( const pid_t pid , const int argc , const char * * argv )
2000-09-13 11:07:17 +04:00
{
2003-06-28 00:55:48 +04:00
const char * cmd ;
2003-04-14 07:59:25 +04:00
/* Check for subcommand */
if ( argc = = 1 ) {
fprintf ( stderr , " Must specify subcommand: \n " ) ;
fprintf ( stderr , " \t queuepause <printername> \n " ) ;
fprintf ( stderr , " \t queueresume <printername> \n " ) ;
fprintf ( stderr , " \t jobpause <printername> <unix jobid> \n " ) ;
fprintf ( stderr , " \t jobresume <printername> <unix jobid> \n " ) ;
fprintf ( stderr , " \t jobdelete <printername> <unix jobid> \n " ) ;
fprintf ( stderr , " \t printer <printername> <comment|port| "
" driver> <value> \n " ) ;
return False ;
2000-09-13 11:07:17 +04:00
}
2003-04-14 07:59:25 +04:00
cmd = argv [ 1 ] ;
2001-06-01 16:04:44 +04:00
2003-04-14 07:59:25 +04:00
if ( strcmp ( cmd , " queuepause " ) = = 0 ) {
2001-02-12 19:18:02 +03:00
2003-04-14 07:59:25 +04:00
if ( argc ! = 3 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> printnotify "
" queuepause <printername> \n " ) ;
return False ;
2002-07-15 14:35:28 +04:00
}
2003-04-14 07:59:25 +04:00
notify_printer_status_byname ( argv [ 2 ] , PRINTER_STATUS_PAUSED ) ;
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
goto send ;
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
} else if ( strcmp ( cmd , " queueresume " ) = = 0 ) {
2000-09-13 11:07:17 +04:00
2003-04-14 07:59:25 +04:00
if ( argc ! = 3 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> printnotify "
" queuereume <printername> \n " ) ;
return False ;
2000-10-07 03:01:47 +04:00
}
2003-04-14 07:59:25 +04:00
notify_printer_status_byname ( argv [ 2 ] , PRINTER_STATUS_OK ) ;
2000-10-07 03:01:47 +04:00
2003-04-14 07:59:25 +04:00
goto send ;
2000-09-13 11:07:17 +04:00
2003-04-14 07:59:25 +04:00
} else if ( strcmp ( cmd , " jobpause " ) = = 0 ) {
int jobid ;
2000-11-11 03:33:33 +03:00
2003-04-14 07:59:25 +04:00
if ( argc ! = 4 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> printnotify "
" jobpause <printername> <unix-jobid> \n " ) ;
return False ;
2002-12-20 23:21:31 +03:00
}
2003-04-14 07:59:25 +04:00
jobid = atoi ( argv [ 3 ] ) ;
notify_job_status_byname (
argv [ 2 ] , jobid , JOB_STATUS_PAUSED ,
SPOOLSS_NOTIFY_MSG_UNIX_JOBID ) ;
goto send ;
} else if ( strcmp ( cmd , " jobresume " ) = = 0 ) {
int jobid ;
if ( argc ! = 4 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> printnotify "
" jobpause <printername> <unix-jobid> \n " ) ;
return False ;
2000-10-07 03:01:47 +04:00
}
2003-04-14 07:59:25 +04:00
jobid = atoi ( argv [ 3 ] ) ;
notify_job_status_byname (
argv [ 2 ] , jobid , JOB_STATUS_QUEUED ,
SPOOLSS_NOTIFY_MSG_UNIX_JOBID ) ;
goto send ;
} else if ( strcmp ( cmd , " jobdelete " ) = = 0 ) {
int jobid ;
if ( argc ! = 4 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> printnotify "
" jobpause <printername> <unix-jobid> \n " ) ;
return False ;
2000-10-12 04:29:01 +04:00
}
2000-10-07 03:01:47 +04:00
2003-04-14 07:59:25 +04:00
jobid = atoi ( argv [ 3 ] ) ;
notify_job_status_byname (
argv [ 2 ] , jobid , JOB_STATUS_DELETING ,
SPOOLSS_NOTIFY_MSG_UNIX_JOBID ) ;
notify_job_status_byname (
argv [ 2 ] , jobid , JOB_STATUS_DELETING |
JOB_STATUS_DELETED ,
SPOOLSS_NOTIFY_MSG_UNIX_JOBID ) ;
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
goto send ;
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
} else if ( strcmp ( cmd , " printer " ) = = 0 ) {
uint32 attribute ;
if ( argc ! = 5 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> printnotify "
" printer <printername> <comment|port|driver> "
" <value> \n " ) ;
return False ;
}
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
if ( strcmp ( argv [ 3 ] , " comment " ) = = 0 ) {
attribute = PRINTER_NOTIFY_COMMENT ;
} else if ( strcmp ( argv [ 3 ] , " port " ) = = 0 ) {
attribute = PRINTER_NOTIFY_PORT_NAME ;
} else if ( strcmp ( argv [ 3 ] , " driver " ) = = 0 ) {
attribute = PRINTER_NOTIFY_DRIVER_NAME ;
} else {
fprintf ( stderr , " Invalid printer command '%s' \n " ,
argv [ 3 ] ) ;
2002-07-15 14:35:28 +04:00
return False ;
2000-11-11 01:07:57 +03:00
}
2002-02-26 06:12:09 +03:00
2003-04-14 07:59:25 +04:00
notify_printer_byname ( argv [ 2 ] , attribute , argv [ 4 ] ) ;
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
goto send ;
}
2002-11-26 03:46:31 +03:00
2003-04-14 07:59:25 +04:00
fprintf ( stderr , " Invalid subcommand '%s' \n " , cmd ) ;
return False ;
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
send :
print_notify_send_messages ( 0 ) ;
return True ;
}
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
/* Close a share */
2002-07-15 14:35:28 +04:00
2003-06-28 00:55:48 +04:00
static BOOL do_closeshare ( const pid_t pid , const int argc , const char * * argv )
2003-04-14 07:59:25 +04:00
{
if ( argc ! = 2 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> close-share "
" <sharename> \n " ) ;
return False ;
}
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
return send_message (
pid , MSG_SMB_FORCE_TDIS , argv [ 1 ] , strlen ( argv [ 1 ] ) + 1 , False ) ;
}
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
/* Force a SAM synchronisation */
2002-07-15 14:35:28 +04:00
2003-06-28 00:55:48 +04:00
static BOOL do_samsync ( const pid_t pid , const int argc , const char * * argv )
2003-04-14 07:59:25 +04:00
{
if ( argc ! = 1 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> samsync \n " ) ;
return False ;
}
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
return send_message (
pid , MSG_SMB_SAM_SYNC , NULL , 0 , False ) ;
}
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
/* Force a SAM replication */
2002-07-15 14:35:28 +04:00
2003-06-28 00:55:48 +04:00
static BOOL do_samrepl ( const pid_t pid , const int argc , const char * * argv )
2003-04-14 07:59:25 +04:00
{
if ( argc ! = 1 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> samrepl \n " ) ;
return False ;
}
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
return send_message (
pid , MSG_SMB_SAM_REPL , NULL , 0 , False ) ;
}
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
/* Display talloc pool usage */
2002-07-15 14:35:28 +04:00
2003-06-28 00:55:48 +04:00
static BOOL do_poolusage ( const pid_t pid , const int argc , const char * * argv )
2003-04-14 07:59:25 +04:00
{
if ( argc ! = 1 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> pool-usage \n " ) ;
return False ;
}
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
/* Send a message and register our interest in a reply */
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
if ( ! send_message ( pid , MSG_REQ_POOL_USAGE , NULL , 0 , False ) )
return False ;
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
message_register ( MSG_POOL_USAGE , print_string_cb ) ;
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
wait_replies ( pid = = 0 ) ;
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
/* No replies were received within the timeout period */
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
if ( num_replies = = 0 )
printf ( " No replies received \n " ) ;
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
message_deregister ( MSG_POOL_USAGE ) ;
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
return num_replies ;
}
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
/* Perform a dmalloc mark */
2002-07-15 14:35:28 +04:00
2003-06-28 00:55:48 +04:00
static BOOL do_dmalloc_mark ( const pid_t pid , const int argc , const char * * argv )
2003-04-14 07:59:25 +04:00
{
if ( argc ! = 1 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> dmalloc-mark \n " ) ;
return False ;
}
2002-07-15 14:35:28 +04:00
2003-04-14 07:59:25 +04:00
return send_message (
pid , MSG_REQ_DMALLOC_MARK , NULL , 0 , False ) ;
}
2002-11-26 03:46:31 +03:00
2003-04-14 07:59:25 +04:00
/* Perform a dmalloc changed */
2000-11-11 01:07:57 +03:00
2003-06-28 00:55:48 +04:00
static BOOL do_dmalloc_changed ( const pid_t pid , const int argc , const char * * argv )
2003-04-14 07:59:25 +04:00
{
if ( argc ! = 1 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> "
" dmalloc-log-changed \n " ) ;
return False ;
}
2001-06-20 07:05:09 +04:00
2003-04-14 07:59:25 +04:00
return send_message (
pid , MSG_REQ_DMALLOC_LOG_CHANGED , NULL , 0 , False ) ;
}
2001-08-28 10:38:11 +04:00
2003-04-14 07:59:25 +04:00
/* Shutdown a server process */
2001-08-28 10:38:11 +04:00
2003-06-28 00:55:48 +04:00
static BOOL do_shutdown ( const pid_t pid , const int argc , const char * * argv )
2003-04-14 07:59:25 +04:00
{
if ( argc ! = 1 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> shutdown \n " ) ;
return False ;
}
2001-08-28 10:38:11 +04:00
2003-04-14 07:59:25 +04:00
return send_message ( pid , MSG_SHUTDOWN , NULL , 0 , False ) ;
}
2001-08-28 10:38:11 +04:00
2003-04-14 07:59:25 +04:00
/* Notify a driver upgrade */
2001-08-28 10:38:11 +04:00
2003-06-28 00:55:48 +04:00
static BOOL do_drvupgrade ( const pid_t pid , const int argc , const char * * argv )
2003-04-14 07:59:25 +04:00
{
if ( argc ! = 2 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> drvupgrade "
" <driver-name> \n " ) ;
return False ;
}
2001-08-28 10:38:11 +04:00
2003-04-14 07:59:25 +04:00
return send_message (
pid , MSG_DEBUG , argv [ 1 ] , strlen ( argv [ 1 ] ) + 1 , False ) ;
}
2001-08-28 10:38:11 +04:00
2003-07-15 21:21:21 +04:00
static BOOL do_reload_config ( const pid_t pid , const int argc , const char * * argv )
{
if ( argc ! = 1 ) {
fprintf ( stderr , " Usage: smbcontrol <dest> reload-config \n " ) ;
return False ;
}
return send_message ( pid , MSG_SMB_CONF_UPDATED , NULL , 0 , False ) ;
}
2003-04-14 07:59:25 +04:00
/* A list of message type supported */
2001-08-28 10:38:11 +04:00
2003-04-14 07:59:25 +04:00
static const struct {
const char * name ; /* Option name */
2003-06-28 00:55:48 +04:00
BOOL ( * fn ) ( const pid_t pid , const int argc , const char * * argv ) ;
2003-04-14 07:59:25 +04:00
const char * help ; /* Short help text */
} msg_types [ ] = {
{ " debug " , do_debug , " Set debuglevel " } ,
{ " force-election " , do_election ,
" Force a browse election " } ,
{ " ping " , do_ping , " Elicit a response " } ,
{ " profile " , do_profile , " " } ,
{ " profilelevel " , do_profilelevel , " " } ,
{ " debuglevel " , do_debuglevel , " Display current debuglevels " } ,
{ " printnotify " , do_printnotify , " Send a print notify message " } ,
{ " close-share " , do_closeshare , " Forcibly disconnect a share " } ,
{ " samsync " , do_samsync , " Initiate SAM synchronisation " } ,
{ " samrepl " , do_samrepl , " Initiate SAM replication " } ,
{ " pool-usage " , do_poolusage , " Display talloc memory usage " } ,
{ " dmalloc-mark " , do_dmalloc_mark , " " } ,
{ " dmalloc-log-changed " , do_dmalloc_changed , " " } ,
{ " shutdown " , do_shutdown , " Shut down daemon " } ,
{ " drvupgrade " , do_drvupgrade , " Notify a printer driver has changed " } ,
2003-07-15 21:21:21 +04:00
{ " reload-config " , do_reload_config , " Force smbd or winbindd to reload config file " } ,
2003-04-14 07:59:25 +04:00
{ " noop " , do_noop , " Do nothing " } ,
{ NULL }
} ;
2001-08-28 10:38:11 +04:00
2003-04-14 07:59:25 +04:00
/* Display usage information */
2002-01-09 11:27:15 +03:00
2003-04-14 07:59:25 +04:00
static void usage ( poptContext * pc )
{
int i ;
2002-03-29 16:49:48 +03:00
2003-04-14 07:59:25 +04:00
poptPrintHelp ( * pc , stderr , 0 ) ;
2003-04-03 19:44:19 +04:00
2003-04-14 07:59:25 +04:00
fprintf ( stderr , " \n " ) ;
fprintf ( stderr , " <destination> is one of \" nmbd \" , \" smbd \" or a "
" process ID \n " ) ;
fprintf ( stderr , " \n " ) ;
fprintf ( stderr , " <message-type> is one of: \n " ) ;
for ( i = 0 ; msg_types [ i ] . name ; i + + )
fprintf ( stderr , " \t %-30s%s \n " , msg_types [ i ] . name ,
msg_types [ i ] . help ) ;
fprintf ( stderr , " \n " ) ;
exit ( 1 ) ;
}
/* Return the pid number for a string destination */
2003-06-28 00:55:48 +04:00
static pid_t parse_dest ( const char * dest )
2003-04-14 07:59:25 +04:00
{
pid_t pid ;
/* Zero is a special return value for broadcast smbd */
if ( strequal ( dest , " smbd " ) )
return 0 ;
/* Try self - useful for testing */
if ( strequal ( dest , " self " ) )
return sys_getpid ( ) ;
/* Check for numeric pid number */
if ( ( pid = atoi ( dest ) ) ! = 0 )
return pid ;
/* Look up other destinations in pidfile directory */
if ( ( pid = pidfile_pid ( dest ) ) ! = 0 )
return pid ;
fprintf ( stderr , " Can't find pid for destination '%s' \n " , dest ) ;
return - 1 ;
}
/* Execute smbcontrol command */
2003-06-28 00:55:48 +04:00
static BOOL do_command ( int argc , const char * * argv )
2003-04-14 07:59:25 +04:00
{
2003-06-28 00:55:48 +04:00
const char * dest = argv [ 0 ] , * command = argv [ 1 ] ;
2003-04-14 07:59:25 +04:00
pid_t pid ;
int i ;
/* Check destination */
if ( ( pid = parse_dest ( dest ) ) = = - 1 )
return False ;
/* Check command */
for ( i = 0 ; msg_types [ i ] . name ; i + + ) {
if ( strequal ( command , msg_types [ i ] . name ) )
return msg_types [ i ] . fn ( pid , argc - 1 , argv + 1 ) ;
2000-09-13 11:07:17 +04:00
}
2001-12-21 07:28:03 +03:00
2003-04-14 07:59:25 +04:00
fprintf ( stderr , " smbcontrol: unknown command '%s' \n " , command ) ;
2002-11-26 03:46:31 +03:00
2003-04-14 07:59:25 +04:00
return False ;
2000-09-30 00:08:00 +04:00
}
2003-04-14 07:59:25 +04:00
/* Main program */
2003-06-28 00:55:48 +04:00
int main ( int argc , const char * * argv )
2000-09-30 00:08:00 +04:00
{
2003-04-14 07:59:25 +04:00
poptContext pc ;
2000-09-30 00:08:00 +04:00
int opt ;
2003-04-30 17:22:02 +04:00
static struct poptOption wbinfo_options [ ] = {
2003-04-14 07:59:25 +04:00
{ " timeout " , ' t ' , POPT_ARG_INT , & timeout , ' t ' ,
" Set timeout value in seconds " , " TIMEOUT " } ,
{ " configfile " , ' s ' , POPT_ARG_STRING , NULL , ' s ' ,
" Use alternative configuration file " , " CONFIGFILE " } ,
POPT_TABLEEND
} ;
struct poptOption options [ ] = {
{ NULL , 0 , POPT_ARG_INCLUDE_TABLE , wbinfo_options , 0 ,
" Options " } ,
POPT_AUTOHELP
POPT_COMMON_VERSION
POPT_TABLEEND
} ;
2002-01-08 00:32:22 +03:00
2000-09-30 00:08:00 +04:00
setup_logging ( argv [ 0 ] , True ) ;
2003-04-14 07:59:25 +04:00
/* Parse command line arguments using popt */
pc = poptGetContext (
" smbcontrol " , argc , ( const char * * ) argv , options , 0 ) ;
poptSetOtherOptionHelp ( pc , " [OPTION...] <destination> <message-type> "
" <parameters> " ) ;
if ( argc = = 1 )
usage ( & pc ) ;
2000-09-30 00:08:00 +04:00
2003-04-14 07:59:25 +04:00
while ( ( opt = poptGetNextOpt ( pc ) ) ! = - 1 ) {
switch ( opt ) {
case ' t ' : /* --timeout */
argc - = 2 ;
2000-09-30 00:08:00 +04:00
break ;
2003-04-14 07:59:25 +04:00
case ' s ' : /* --configfile */
2001-11-19 05:49:53 +03:00
pstrcpy ( dyn_CONFIGFILE , optarg ) ;
2003-04-14 07:59:25 +04:00
argc - = 2 ;
2001-09-21 17:37:52 +04:00
break ;
2000-09-30 00:08:00 +04:00
default :
2003-04-14 07:59:25 +04:00
fprintf ( stderr , " Invalid option \n " ) ;
poptPrintHelp ( pc , stderr , 0 ) ;
break ;
2000-09-30 00:08:00 +04:00
}
}
2003-04-14 07:59:25 +04:00
/* We should now have the remaining command line arguments in
argv . The argc parameter should have been decremented to the
correct value in the above switch statement . */
2001-09-21 17:37:52 +04:00
2003-06-28 00:55:48 +04:00
argv = ( const char * * ) poptGetArgs ( pc ) ;
2003-04-14 07:59:25 +04:00
argc - - ; /* Don't forget about argv[0] */
2000-09-30 00:08:00 +04:00
2003-04-14 07:59:25 +04:00
if ( argc = = 1 )
usage ( & pc ) ;
2002-01-09 11:27:15 +03:00
2003-04-14 07:59:25 +04:00
lp_load ( dyn_CONFIGFILE , False , False , False ) ;
2000-09-30 00:08:00 +04:00
2003-04-14 07:59:25 +04:00
/* Need to invert sense of return code -- samba
* routines mostly return True = = 1 for success , but
* shell needs 0. */
return ! do_command ( argc , argv ) ;
2000-09-13 11:07:17 +04:00
}