2005-01-30 13:24:36 +03:00
/*
Unix SMB / CIFS implementation .
NBT interface handling
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
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"
# include "dlinklist.h"
# include "nbt_server/nbt_server.h"
# include "smbd/service_task.h"
2005-01-31 04:57:58 +03:00
2005-02-02 13:29:50 +03:00
/*
receive an incoming request and dispatch it to the right place
*/
2005-02-04 04:39:10 +03:00
static void nbtd_request_handler ( struct nbt_name_socket * nbtsock ,
struct nbt_name_packet * packet ,
const char * src_address , int src_port )
2005-02-02 13:29:50 +03:00
{
2005-02-06 11:25:53 +03:00
/* if its a WINS query then direct to our WINS server if we
are running one */
2005-02-04 04:55:50 +03:00
if ( ( packet - > operation & NBT_FLAG_RECURSION_DESIRED ) & &
2005-02-06 11:25:53 +03:00
! ( packet - > operation & NBT_FLAG_BROADCAST ) & &
lp_wins_support ( ) ) {
2005-02-04 04:55:50 +03:00
nbtd_query_wins ( nbtsock , packet , src_address , src_port ) ;
return ;
}
2005-02-04 05:05:27 +03:00
/* see if its from one of our own interfaces - if so, then ignore it */
if ( nbtd_self_packet ( nbtsock , packet , src_address , src_port ) ) {
DEBUG ( 10 , ( " Ignoring self packet from %s:%d \n " , src_address , src_port ) ) ;
return ;
}
2005-02-04 04:55:50 +03:00
/* the request is to us in our role as a B node */
2005-02-02 13:29:50 +03:00
switch ( packet - > operation & NBT_OPCODE ) {
case NBT_OPCODE_QUERY :
2005-02-04 04:39:10 +03:00
nbtd_request_query ( nbtsock , packet , src_address , src_port ) ;
2005-02-04 04:55:50 +03:00
break ;
case NBT_OPCODE_REGISTER :
case NBT_OPCODE_REFRESH :
nbtd_request_defense ( nbtsock , packet , src_address , src_port ) ;
2005-02-02 13:29:50 +03:00
break ;
2005-02-06 11:25:53 +03:00
default :
nbtd_bad_packet ( packet , src_address , " Unexpected opcode " ) ;
break ;
2005-02-02 13:29:50 +03:00
}
}
2005-01-31 04:57:58 +03:00
/*
find a registered name on an interface
*/
2005-02-06 11:25:53 +03:00
struct nbtd_iface_name * nbtd_find_iname ( struct nbtd_interface * iface ,
struct nbt_name * name ,
uint16_t nb_flags )
2005-01-31 04:57:58 +03:00
{
2005-02-06 11:25:53 +03:00
struct nbtd_iface_name * iname ;
2005-01-31 04:57:58 +03:00
for ( iname = iface - > names ; iname ; iname = iname - > next ) {
if ( iname - > name . type = = name - > type & &
StrCaseCmp ( name - > name , iname - > name . name ) = = 0 & &
( ( iname - > nb_flags & nb_flags ) = = nb_flags ) ) {
return iname ;
}
}
return NULL ;
}
2005-01-30 13:24:36 +03:00
/*
start listening on the given address
*/
2005-02-06 11:25:53 +03:00
static NTSTATUS nbtd_add_socket ( struct nbtd_server * nbtsrv ,
2005-02-04 04:39:10 +03:00
const char * bind_address ,
const char * address ,
const char * bcast ,
const char * netmask )
2005-01-30 13:24:36 +03:00
{
2005-02-06 11:25:53 +03:00
struct nbtd_interface * iface ;
2005-01-30 13:24:36 +03:00
NTSTATUS status ;
2005-02-02 13:29:50 +03:00
struct nbt_name_socket * bcast_nbtsock ;
/*
we actually create two sockets . One listens on the broadcast address
for the interface , and the other listens on our specific address . This
allows us to run with " bind interfaces only " while still receiving
broadcast addresses , and also simplifies matching incoming requests
to interfaces
*/
2005-01-30 13:24:36 +03:00
2005-02-06 11:25:53 +03:00
iface = talloc ( nbtsrv , struct nbtd_interface ) ;
2005-01-30 13:24:36 +03:00
NT_STATUS_HAVE_NO_MEMORY ( iface ) ;
2005-01-31 04:57:58 +03:00
iface - > nbtsrv = nbtsrv ;
2005-01-30 13:24:36 +03:00
iface - > bcast_address = talloc_steal ( iface , bcast ) ;
2005-01-31 04:57:58 +03:00
iface - > ip_address = talloc_steal ( iface , address ) ;
iface - > netmask = talloc_steal ( iface , netmask ) ;
iface - > names = NULL ;
2005-01-30 13:24:36 +03:00
2005-02-02 13:29:50 +03:00
if ( strcmp ( netmask , " 0.0.0.0 " ) ! = 0 ) {
bcast_nbtsock = nbt_name_socket_init ( iface , nbtsrv - > task - > event_ctx ) ;
2005-02-03 14:30:36 +03:00
NT_STATUS_HAVE_NO_MEMORY ( bcast_nbtsock ) ;
2005-02-02 13:29:50 +03:00
status = socket_listen ( bcast_nbtsock - > sock , bcast , lp_nbt_port ( ) , 0 , 0 ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " Failed to bind to %s:%d - %s \n " ,
bcast , lp_nbt_port ( ) , nt_errstr ( status ) ) ) ;
talloc_free ( iface ) ;
return status ;
}
2005-02-04 04:39:10 +03:00
nbt_set_incoming_handler ( bcast_nbtsock , nbtd_request_handler , iface ) ;
2005-02-02 13:29:50 +03:00
}
2005-01-30 13:24:36 +03:00
iface - > nbtsock = nbt_name_socket_init ( iface , nbtsrv - > task - > event_ctx ) ;
NT_STATUS_HAVE_NO_MEMORY ( iface - > ip_address ) ;
2005-02-02 13:35:25 +03:00
status = socket_listen ( iface - > nbtsock - > sock , bind_address , lp_nbt_port ( ) , 0 , 0 ) ;
2005-01-30 13:24:36 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " Failed to bind to %s:%d - %s \n " ,
address , lp_nbt_port ( ) , nt_errstr ( status ) ) ) ;
talloc_free ( iface ) ;
return status ;
}
2005-02-04 04:39:10 +03:00
nbt_set_incoming_handler ( iface - > nbtsock , nbtd_request_handler , iface ) ;
2005-02-02 13:29:50 +03:00
2005-01-31 04:57:58 +03:00
if ( strcmp ( netmask , " 0.0.0.0 " ) = = 0 ) {
DLIST_ADD ( nbtsrv - > bcast_interface , iface ) ;
} else {
DLIST_ADD ( nbtsrv - > interfaces , iface ) ;
}
2005-01-30 13:24:36 +03:00
return NT_STATUS_OK ;
}
2005-02-06 11:25:53 +03:00
/*
setup a socket for talking to our WINS servers
*/
static NTSTATUS nbtd_add_wins_socket ( struct nbtd_server * nbtsrv )
{
struct nbtd_interface * iface ;
iface = talloc_zero ( nbtsrv , struct nbtd_interface ) ;
NT_STATUS_HAVE_NO_MEMORY ( iface ) ;
iface - > nbtsrv = nbtsrv ;
iface - > nbtsock = nbt_name_socket_init ( iface , nbtsrv - > task - > event_ctx ) ;
NT_STATUS_HAVE_NO_MEMORY ( iface - > nbtsock ) ;
DLIST_ADD ( nbtsrv - > wins_interface , iface ) ;
return NT_STATUS_OK ;
}
2005-01-30 13:24:36 +03:00
/*
setup our listening sockets on the configured network interfaces
*/
2005-02-06 11:25:53 +03:00
NTSTATUS nbtd_startup_interfaces ( struct nbtd_server * nbtsrv )
2005-01-30 13:24:36 +03:00
{
int num_interfaces = iface_count ( ) ;
int i ;
TALLOC_CTX * tmp_ctx = talloc_new ( nbtsrv ) ;
NTSTATUS status ;
2005-01-31 04:57:58 +03:00
2005-02-02 13:29:50 +03:00
/* if we are allowing incoming packets from any address, then
we also need to bind to the wildcard address */
if ( ! lp_bind_interfaces_only ( ) ) {
const char * primary_address ;
/* the primary address is the address we will return
for non - WINS queries not made on a specific
interface */
if ( num_interfaces > 0 ) {
primary_address = sys_inet_ntoa ( * iface_n_ip ( 0 ) ) ;
} else {
primary_address = sys_inet_ntoa ( interpret_addr2 (
lp_netbios_name ( ) ) ) ;
}
primary_address = talloc_strdup ( tmp_ctx , primary_address ) ;
NT_STATUS_HAVE_NO_MEMORY ( primary_address ) ;
2005-01-30 13:24:36 +03:00
2005-02-04 04:39:10 +03:00
status = nbtd_add_socket ( nbtsrv ,
" 0.0.0.0 " ,
primary_address ,
talloc_strdup ( tmp_ctx , " 255.255.255.255 " ) ,
talloc_strdup ( tmp_ctx , " 0.0.0.0 " ) ) ;
2005-02-02 13:29:50 +03:00
NT_STATUS_NOT_OK_RETURN ( status ) ;
}
2005-01-30 13:24:36 +03:00
for ( i = 0 ; i < num_interfaces ; i + + ) {
const char * address = talloc_strdup ( tmp_ctx , sys_inet_ntoa ( * iface_n_ip ( i ) ) ) ;
const char * bcast = talloc_strdup ( tmp_ctx , sys_inet_ntoa ( * iface_n_bcast ( i ) ) ) ;
2005-01-31 04:57:58 +03:00
const char * netmask = talloc_strdup ( tmp_ctx , sys_inet_ntoa ( * iface_n_netmask ( i ) ) ) ;
2005-01-30 13:24:36 +03:00
2005-02-04 04:39:10 +03:00
status = nbtd_add_socket ( nbtsrv , address , address , bcast , netmask ) ;
2005-01-30 13:24:36 +03:00
NT_STATUS_NOT_OK_RETURN ( status ) ;
}
2005-02-06 11:25:53 +03:00
if ( lp_wins_server_list ( ) ) {
status = nbtd_add_wins_socket ( nbtsrv ) ;
NT_STATUS_NOT_OK_RETURN ( status ) ;
}
2005-01-30 13:24:36 +03:00
talloc_free ( tmp_ctx ) ;
return NT_STATUS_OK ;
}
2005-02-06 11:25:53 +03:00
/*
form a list of addresses that we should use in name query replies
2005-02-07 14:49:55 +03:00
we always place the IP in the given interface first
2005-02-06 11:25:53 +03:00
*/
2005-02-07 14:49:55 +03:00
const char * * nbtd_address_list ( struct nbtd_interface * iface , TALLOC_CTX * mem_ctx )
2005-02-06 11:25:53 +03:00
{
2005-02-07 14:49:55 +03:00
struct nbtd_server * nbtsrv = iface - > nbtsrv ;
2005-02-06 11:25:53 +03:00
const char * * ret = NULL ;
2005-02-07 14:49:55 +03:00
struct nbtd_interface * iface2 ;
int count ;
ret = talloc_array ( mem_ctx , const char * , 2 ) ;
if ( ret = = NULL ) goto failed ;
ret [ 0 ] = talloc_strdup ( ret , iface - > ip_address ) ;
if ( ret [ 0 ] = = NULL ) goto failed ;
ret [ 1 ] = NULL ;
count = 1 ;
for ( iface2 = nbtsrv - > interfaces ; iface2 ; iface2 = iface2 - > next ) {
const char * * ret2 ;
if ( strcmp ( iface2 - > ip_address , iface - > ip_address ) = = 0 ) {
continue ;
}
2005-02-06 11:25:53 +03:00
2005-02-07 14:49:55 +03:00
ret2 = talloc_realloc ( mem_ctx , ret , const char * , count + 2 ) ;
2005-02-06 11:25:53 +03:00
if ( ret2 = = NULL ) goto failed ;
ret = ret2 ;
2005-02-07 14:49:55 +03:00
ret [ count ] = talloc_strdup ( ret , iface2 - > ip_address ) ;
2005-02-06 11:25:53 +03:00
if ( ret [ count ] = = NULL ) goto failed ;
count + + ;
}
ret [ count ] = NULL ;
return ret ;
failed :
talloc_free ( ret ) ;
return NULL ;
}