2005-01-30 13:24:36 +03:00
/*
Unix SMB / CIFS implementation .
NBT server task
Copyright ( C ) Andrew Tridgell 2005
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 06:07:03 +04:00
the Free Software Foundation ; either version 3 of the License , or
2005-01-30 13:24:36 +03: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 06:07:03 +04:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2005-01-30 13:24:36 +03:00
*/
# include "includes.h"
2020-11-20 17:27:17 +03:00
# include "samba/service_task.h"
# include "samba/service.h"
2005-01-30 13:24:36 +03:00
# include "nbt_server/nbt_server.h"
2005-12-30 15:43:11 +03:00
# include "nbt_server/wins/winsserver.h"
2006-03-07 14:07:23 +03:00
# include "system/network.h"
2006-08-17 17:37:04 +04:00
# include "lib/socket/netif.h"
2006-12-13 14:19:51 +03:00
# include "auth/auth.h"
# include "dsdb/samdb/samdb.h"
2007-12-03 19:41:50 +03:00
# include "param/param.h"
2024-02-14 14:34:48 +03:00
# include "dynconfig/dynconfig.h"
# include "lib/util/pidfile.h"
# include "lib/util/util_net.h"
# include "lib/socket/socket.h"
# include "../source3/include/fstring.h"
# include "../source3/libsmb/nmblib.h"
# include "../source3/libsmb/unexpected.h"
# include "../source3/lib/util_procid.h"
2005-01-30 13:24:36 +03:00
2017-04-20 22:24:43 +03:00
NTSTATUS server_service_nbtd_init ( TALLOC_CTX * ) ;
2011-03-19 02:45:45 +03:00
2024-02-14 14:34:48 +03:00
static void nbtd_server_msg_send_packet ( struct imessaging_context * msg ,
void * private_data ,
uint32_t msg_type ,
struct server_id src ,
size_t num_fds ,
int * fds ,
DATA_BLOB * data )
{
TALLOC_CTX * frame = talloc_stackframe ( ) ;
struct nbtd_server * nbtsrv =
talloc_get_type_abort ( private_data ,
struct nbtd_server ) ;
struct packet_struct * p = ( struct packet_struct * ) data - > data ;
struct sockaddr_storage ss ;
struct socket_address * dst = NULL ;
struct nbtd_interface * iface = NULL ;
char buf [ 1024 ] = { 0 , } ;
DATA_BLOB blob = { . length = 0 , } ;
DBG_DEBUG ( " Received send_packet from %u \n " , ( unsigned int ) procid_to_pid ( & src ) ) ;
if ( data - > length ! = sizeof ( struct packet_struct ) ) {
DBG_WARNING ( " Discarding invalid packet length from %u \n " ,
( unsigned int ) procid_to_pid ( & src ) ) ;
TALLOC_FREE ( frame ) ;
return ;
}
if ( ( p - > packet_type ! = NMB_PACKET ) & &
( p - > packet_type ! = DGRAM_PACKET ) ) {
DBG_WARNING ( " Discarding invalid packet type from %u: %d \n " ,
( unsigned int ) procid_to_pid ( & src ) , p - > packet_type ) ;
TALLOC_FREE ( frame ) ;
return ;
}
if ( p - > packet_type = = DGRAM_PACKET ) {
p - > port = 138 ;
}
in_addr_to_sockaddr_storage ( & ss , p - > ip ) ;
dst = socket_address_from_sockaddr_storage ( frame , & ss , p - > port ) ;
if ( dst = = NULL ) {
TALLOC_FREE ( frame ) ;
return ;
}
if ( p - > port = = 0 ) {
DBG_WARNING ( " Discarding packet with missing port for addr[%s] "
" from %u \n " ,
dst - > addr , ( unsigned int ) procid_to_pid ( & src ) ) ;
TALLOC_FREE ( frame ) ;
return ;
}
iface = nbtd_find_request_iface ( nbtsrv , dst - > addr , true ) ;
if ( iface = = NULL ) {
DBG_WARNING ( " Could not find iface for packet to addr[%s] "
" from %u \n " ,
dst - > addr , ( unsigned int ) procid_to_pid ( & src ) ) ;
TALLOC_FREE ( frame ) ;
return ;
}
p - > recv_fd = - 1 ;
p - > send_fd = - 1 ;
if ( p - > packet_type = = DGRAM_PACKET ) {
p - > packet . dgram . header . source_ip . s_addr = interpret_addr ( iface - > ip_address ) ;
p - > packet . dgram . header . source_port = 138 ;
}
blob . length = build_packet ( buf , sizeof ( buf ) , p ) ;
if ( blob . length = = 0 ) {
TALLOC_FREE ( frame ) ;
return ;
}
blob . data = ( uint8_t * ) buf ;
if ( p - > packet_type = = DGRAM_PACKET ) {
nbt_dgram_send_raw ( iface - > dgmsock , dst , blob ) ;
} else {
nbt_name_send_raw ( iface - > nbtsock , dst , blob ) ;
}
TALLOC_FREE ( frame ) ;
}
static int nbtd_server_destructor ( struct nbtd_server * nbtsrv )
{
struct task_server * task = nbtsrv - > task ;
pidfile_unlink ( lpcfg_pid_directory ( task - > lp_ctx ) , " nmbd " ) ;
return 0 ;
}
2005-01-30 13:24:36 +03:00
/*
startup the nbtd task
*/
2018-08-23 00:35:52 +03:00
static NTSTATUS nbtd_task_init ( struct task_server * task )
2005-01-30 13:24:36 +03:00
{
2005-02-06 11:25:53 +03:00
struct nbtd_server * nbtsrv ;
2005-01-30 13:41:15 +03:00
NTSTATUS status ;
2007-12-12 00:23:14 +03:00
struct interface * ifaces ;
2024-02-14 14:34:48 +03:00
const char * nmbd_socket_dir = NULL ;
int unexpected_clients ;
2005-01-30 13:24:36 +03:00
2011-06-02 09:40:28 +04:00
load_interface_list ( task , task - > lp_ctx , & ifaces ) ;
2007-12-12 00:23:14 +03:00
2011-05-02 09:57:19 +04:00
if ( iface_list_count ( ifaces ) = = 0 ) {
2009-09-19 05:05:55 +04:00
task_server_terminate ( task , " nbtd: no network interfaces configured " , false ) ;
2018-08-23 00:35:52 +03:00
return NT_STATUS_UNSUCCESSFUL ;
2005-02-16 04:48:11 +03:00
}
2013-02-06 13:58:18 +04:00
if ( lpcfg_disable_netbios ( task - > lp_ctx ) ) {
task_server_terminate ( task , " nbtd: 'disable netbios = yes' set in smb.conf, shutting down nbt server " , false ) ;
2018-08-23 00:35:52 +03:00
return NT_STATUS_UNSUCCESSFUL ;
2013-02-06 13:58:18 +04:00
}
2006-03-09 20:48:41 +03:00
task_server_set_title ( task , " task[nbtd] " ) ;
2005-02-06 11:25:53 +03:00
nbtsrv = talloc ( task , struct nbtd_server ) ;
2005-01-30 13:24:36 +03:00
if ( nbtsrv = = NULL ) {
2009-09-19 05:05:55 +04:00
task_server_terminate ( task , " nbtd: out of memory " , true ) ;
2018-08-23 00:35:52 +03:00
return NT_STATUS_NO_MEMORY ;
2005-01-30 13:24:36 +03:00
}
2005-01-31 04:57:58 +03:00
nbtsrv - > task = task ;
nbtsrv - > interfaces = NULL ;
nbtsrv - > bcast_interface = NULL ;
2005-02-06 11:25:53 +03:00
nbtsrv - > wins_interface = NULL ;
2005-01-30 13:24:36 +03:00
2024-02-14 14:34:48 +03:00
talloc_set_destructor ( nbtsrv , nbtd_server_destructor ) ;
2005-01-30 13:41:15 +03:00
/* start listening on the configured network interfaces */
2007-12-12 00:23:14 +03:00
status = nbtd_startup_interfaces ( nbtsrv , task - > lp_ctx , ifaces ) ;
2005-01-30 13:41:15 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-09-19 05:05:55 +04:00
task_server_terminate ( task , " nbtd failed to setup interfaces " , true ) ;
2018-08-23 00:35:52 +03:00
return status ;
2005-01-30 13:41:15 +03:00
}
2005-01-30 13:24:36 +03:00
2024-02-14 14:34:48 +03:00
nmbd_socket_dir = lpcfg_parm_string ( task - > lp_ctx ,
NULL ,
" nmbd " ,
" socket dir " ) ;
if ( nmbd_socket_dir = = NULL ) {
nmbd_socket_dir = get_dyn_NMBDSOCKETDIR ( ) ;
}
unexpected_clients = lpcfg_parm_int ( task - > lp_ctx ,
NULL ,
" nmbd " ,
" unexpected_clients " ,
200 ) ;
status = nb_packet_server_create ( nbtsrv ,
nbtsrv - > task - > event_ctx ,
nmbd_socket_dir ,
unexpected_clients ,
& nbtsrv - > unexpected_server ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
task_server_terminate ( task , " nbtd failed to start unexpected_server " , true ) ;
return status ;
}
2018-04-11 21:41:30 +03:00
nbtsrv - > sam_ctx = samdb_connect ( nbtsrv ,
task - > event_ctx ,
task - > lp_ctx ,
system_session ( task - > lp_ctx ) ,
NULL ,
0 ) ;
2006-12-13 14:19:51 +03:00
if ( nbtsrv - > sam_ctx = = NULL ) {
2009-09-19 05:05:55 +04:00
task_server_terminate ( task , " nbtd failed to open samdb " , true ) ;
2018-08-23 00:35:52 +03:00
return NT_STATUS_UNSUCCESSFUL ;
2006-12-13 14:19:51 +03:00
}
2005-02-12 02:54:37 +03:00
/* start the WINS server, if appropriate */
status = nbtd_winsserver_init ( nbtsrv ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-09-19 05:05:55 +04:00
task_server_terminate ( task , " nbtd failed to start WINS server " , true ) ;
2018-08-23 00:35:52 +03:00
return status ;
2005-02-12 02:54:37 +03:00
}
2005-10-04 05:51:55 +04:00
nbtd_register_irpc ( nbtsrv ) ;
2005-09-26 01:01:56 +04:00
2024-02-14 14:34:48 +03:00
status = imessaging_register ( task - > msg_ctx ,
nbtsrv ,
MSG_SEND_PACKET ,
nbtd_server_msg_send_packet ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
task_server_terminate ( task , " nbtd failed imessaging_register(MSG_SEND_PACKET) " , true ) ;
return status ;
}
2005-01-31 04:57:58 +03:00
/* start the process of registering our names on all interfaces */
2005-02-04 04:39:10 +03:00
nbtd_register_names ( nbtsrv ) ;
2005-07-10 05:08:10 +04:00
irpc_add_name ( task - > msg_ctx , " nbt_server " ) ;
2018-08-23 00:35:52 +03:00
2024-02-14 14:34:48 +03:00
pidfile_create ( lpcfg_pid_directory ( task - > lp_ctx ) , " nmbd " ) ;
2018-08-23 00:35:52 +03:00
return NT_STATUS_OK ;
2005-01-30 13:24:36 +03:00
}
/*
register ourselves as a available server
*/
2017-04-20 22:24:43 +03:00
NTSTATUS server_service_nbtd_init ( TALLOC_CTX * ctx )
2005-01-30 13:24:36 +03:00
{
2018-08-23 00:29:56 +03:00
static const struct service_details details = {
2017-09-14 22:09:23 +03:00
. inhibit_fork_on_accept = true ,
2018-08-23 00:35:52 +03:00
. inhibit_pre_fork = true ,
. task_init = nbtd_task_init ,
. post_fork = NULL
2017-09-14 22:09:23 +03:00
} ;
2018-08-23 00:35:52 +03:00
return register_server_service ( ctx , " nbt " , & details ) ;
2005-01-30 13:24:36 +03:00
}