2005-01-30 10:24:36 +00: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
2007-07-10 02:07:03 +00:00
the Free Software Foundation ; either version 3 of the License , or
2005-01-30 10:24:36 +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/>.
2005-01-30 10:24:36 +00:00
*/
# include "includes.h"
2006-08-30 11:29:34 +00:00
# include "lib/util/dlinklist.h"
2005-01-30 10:24:36 +00:00
# include "nbt_server/nbt_server.h"
# include "smbd/service_task.h"
2005-02-10 06:59:29 +00:00
# include "lib/socket/socket.h"
2005-12-30 12:43:11 +00:00
# include "nbt_server/wins/winsserver.h"
2006-03-07 11:07:23 +00:00
# include "nbt_server/dgram/proto.h"
# include "system/network.h"
2006-08-17 13:37:04 +00:00
# include "lib/socket/netif.h"
2005-01-31 01:57:58 +00:00
2005-02-02 10:29:50 +00:00
/*
receive an incoming request and dispatch it to the right place
*/
2005-02-04 01:39:10 +00:00
static void nbtd_request_handler ( struct nbt_name_socket * nbtsock ,
struct nbt_name_packet * packet ,
2006-01-09 22:12:53 +00:00
struct socket_address * src )
2005-02-02 10:29:50 +00:00
{
2005-07-10 08:41:02 +00:00
struct nbtd_interface * iface = talloc_get_type ( nbtsock - > incoming . private ,
struct nbtd_interface ) ;
struct nbtd_server * nbtsrv = iface - > nbtsrv ;
nbtsrv - > stats . total_received + + ;
2005-02-04 02:05:27 +00:00
/* see if its from one of our own interfaces - if so, then ignore it */
2006-01-18 16:36:53 +00:00
if ( nbtd_self_packet_and_bcast ( nbtsock , packet , src ) ) {
DEBUG ( 10 , ( " Ignoring bcast self packet from %s:%d \n " , src - > addr , src - > port ) ) ;
2005-02-04 02:05:27 +00:00
return ;
}
2005-02-11 23:54:37 +00:00
switch ( packet - > operation & NBT_OPCODE ) {
2005-02-02 10:29:50 +00:00
case NBT_OPCODE_QUERY :
2005-07-10 08:41:02 +00:00
nbtsrv - > stats . query_count + + ;
2005-10-14 12:22:15 +00:00
nbtd_request_query ( nbtsock , packet , src ) ;
2005-02-04 01:55:50 +00:00
break ;
case NBT_OPCODE_REGISTER :
case NBT_OPCODE_REFRESH :
2005-02-08 01:09:21 +00:00
case NBT_OPCODE_REFRESH2 :
2005-07-10 08:41:02 +00:00
nbtsrv - > stats . register_count + + ;
2005-10-14 12:22:15 +00:00
nbtd_request_defense ( nbtsock , packet , src ) ;
2005-02-02 10:29:50 +00:00
break ;
2005-02-06 08:25:53 +00:00
2005-02-11 23:54:37 +00:00
case NBT_OPCODE_RELEASE :
case NBT_OPCODE_MULTI_HOME_REG :
2005-07-10 08:41:02 +00:00
nbtsrv - > stats . release_count + + ;
2005-10-14 12:22:15 +00:00
nbtd_winsserver_request ( nbtsock , packet , src ) ;
2005-02-11 23:54:37 +00:00
break ;
2005-02-06 08:25:53 +00:00
default :
2005-10-14 12:22:15 +00:00
nbtd_bad_packet ( packet , src , " Unexpected opcode " ) ;
2005-02-06 08:25:53 +00:00
break ;
2005-02-02 10:29:50 +00:00
}
}
2005-01-31 01:57:58 +00:00
/*
find a registered name on an interface
*/
2005-02-06 08:25:53 +00:00
struct nbtd_iface_name * nbtd_find_iname ( struct nbtd_interface * iface ,
struct nbt_name * name ,
uint16_t nb_flags )
2005-01-31 01:57:58 +00:00
{
2005-02-06 08:25:53 +00:00
struct nbtd_iface_name * iname ;
2005-01-31 01:57:58 +00:00
for ( iname = iface - > names ; iname ; iname = iname - > next ) {
if ( iname - > name . type = = name - > type & &
2005-02-11 08:18:55 +00:00
strcmp ( name - > name , iname - > name . name ) = = 0 & &
2005-01-31 01:57:58 +00:00
( ( iname - > nb_flags & nb_flags ) = = nb_flags ) ) {
return iname ;
}
}
return NULL ;
}
2005-01-30 10:24:36 +00:00
/*
start listening on the given address
*/
2005-02-06 08:25:53 +00:00
static NTSTATUS nbtd_add_socket ( struct nbtd_server * nbtsrv ,
2005-02-04 01:39:10 +00:00
const char * bind_address ,
const char * address ,
const char * bcast ,
const char * netmask )
2005-01-30 10:24:36 +00:00
{
2005-02-06 08:25:53 +00:00
struct nbtd_interface * iface ;
2005-01-30 10:24:36 +00:00
NTSTATUS status ;
2006-01-09 22:12:53 +00:00
struct socket_address * bcast_address ;
struct socket_address * unicast_address ;
2005-02-02 10:29:50 +00:00
/*
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 10:24:36 +00:00
2005-02-06 08:25:53 +00:00
iface = talloc ( nbtsrv , struct nbtd_interface ) ;
2005-01-30 10:24:36 +00:00
NT_STATUS_HAVE_NO_MEMORY ( iface ) ;
2005-01-31 01:57:58 +00:00
iface - > nbtsrv = nbtsrv ;
2005-01-30 10:24:36 +00:00
iface - > bcast_address = talloc_steal ( iface , bcast ) ;
2005-01-31 01:57:58 +00:00
iface - > ip_address = talloc_steal ( iface , address ) ;
iface - > netmask = talloc_steal ( iface , netmask ) ;
iface - > names = NULL ;
2005-01-30 10:24:36 +00:00
2005-02-02 10:29:50 +00:00
if ( strcmp ( netmask , " 0.0.0.0 " ) ! = 0 ) {
2005-04-03 04:32:37 +00:00
struct nbt_name_socket * bcast_nbtsock ;
/* listen for broadcasts on port 137 */
2005-02-02 10:29:50 +00:00
bcast_nbtsock = nbt_name_socket_init ( iface , nbtsrv - > task - > event_ctx ) ;
2006-01-09 22:12:53 +00:00
if ( ! bcast_nbtsock ) {
talloc_free ( iface ) ;
return NT_STATUS_NO_MEMORY ;
}
bcast_address = socket_address_from_strings ( bcast_nbtsock , bcast_nbtsock - > sock - > backend_name ,
bcast , lp_nbt_port ( ) ) ;
if ( ! bcast_address ) {
talloc_free ( iface ) ;
return NT_STATUS_NO_MEMORY ;
}
2005-02-02 10:29:50 +00:00
2006-01-09 22:12:53 +00:00
status = socket_listen ( bcast_nbtsock - > sock , bcast_address , 0 , 0 ) ;
2005-02-02 10:29:50 +00:00
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 ;
}
2006-01-09 22:12:53 +00:00
talloc_free ( bcast_address ) ;
2005-02-02 10:29:50 +00:00
2005-02-04 01:39:10 +00:00
nbt_set_incoming_handler ( bcast_nbtsock , nbtd_request_handler , iface ) ;
2005-02-02 10:29:50 +00:00
}
2005-04-03 04:32:37 +00:00
/* listen for unicasts on port 137 */
2005-01-30 10:24:36 +00:00
iface - > nbtsock = nbt_name_socket_init ( iface , nbtsrv - > task - > event_ctx ) ;
2006-01-09 22:12:53 +00:00
if ( ! iface - > nbtsock ) {
talloc_free ( iface ) ;
return NT_STATUS_NO_MEMORY ;
}
2005-01-30 10:24:36 +00:00
2006-01-09 22:12:53 +00:00
unicast_address = socket_address_from_strings ( iface - > nbtsock , iface - > nbtsock - > sock - > backend_name ,
bind_address , lp_nbt_port ( ) ) ;
status = socket_listen ( iface - > nbtsock - > sock , unicast_address , 0 , 0 ) ;
2005-01-30 10:24:36 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " Failed to bind to %s:%d - %s \n " ,
2005-04-08 08:57:09 +00:00
bind_address , lp_nbt_port ( ) , nt_errstr ( status ) ) ) ;
2005-01-30 10:24:36 +00:00
talloc_free ( iface ) ;
return status ;
}
2006-01-09 22:12:53 +00:00
talloc_free ( unicast_address ) ;
2005-02-04 01:39:10 +00:00
nbt_set_incoming_handler ( iface - > nbtsock , nbtd_request_handler , iface ) ;
2005-02-02 10:29:50 +00:00
2005-04-08 08:57:09 +00:00
/* also setup the datagram listeners */
status = nbtd_dgram_setup ( iface , bind_address ) ;
2005-04-03 04:32:37 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2005-04-08 08:57:09 +00:00
DEBUG ( 0 , ( " Failed to setup dgram listen on %s - %s \n " ,
bind_address , nt_errstr ( status ) ) ) ;
2005-04-03 04:32:37 +00:00
talloc_free ( iface ) ;
return status ;
}
2005-04-08 08:57:09 +00:00
2005-01-31 01:57:58 +00:00
if ( strcmp ( netmask , " 0.0.0.0 " ) = = 0 ) {
DLIST_ADD ( nbtsrv - > bcast_interface , iface ) ;
} else {
DLIST_ADD ( nbtsrv - > interfaces , iface ) ;
}
2005-01-30 10:24:36 +00:00
return NT_STATUS_OK ;
}
2005-02-06 08:25:53 +00: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 ;
DLIST_ADD ( nbtsrv - > wins_interface , iface ) ;
return NT_STATUS_OK ;
}
2005-01-30 10:24:36 +00:00
/*
setup our listening sockets on the configured network interfaces
*/
2005-02-06 08:25:53 +00:00
NTSTATUS nbtd_startup_interfaces ( struct nbtd_server * nbtsrv )
2005-01-30 10:24:36 +00:00
{
int num_interfaces = iface_count ( ) ;
int i ;
TALLOC_CTX * tmp_ctx = talloc_new ( nbtsrv ) ;
NTSTATUS status ;
2005-01-31 01:57:58 +00:00
2005-02-02 10:29:50 +00: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 ) {
2005-02-10 03:22:47 +00:00
primary_address = iface_n_ip ( 0 ) ;
2005-02-02 10:29:50 +00:00
} 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 10:24:36 +00:00
2005-02-04 01:39:10 +00: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 10:29:50 +00:00
NT_STATUS_NOT_OK_RETURN ( status ) ;
}
2005-01-30 10:24:36 +00:00
for ( i = 0 ; i < num_interfaces ; i + + ) {
2006-02-15 04:18:11 +00:00
const char * bcast = iface_n_bcast ( i ) ;
const char * address , * netmask ;
/* we can't assume every interface is broadcast capable */
if ( bcast = = NULL ) continue ;
address = talloc_strdup ( tmp_ctx , iface_n_ip ( i ) ) ;
bcast = talloc_strdup ( tmp_ctx , bcast ) ;
netmask = talloc_strdup ( tmp_ctx , iface_n_netmask ( i ) ) ;
2005-01-30 10:24:36 +00:00
2005-02-04 01:39:10 +00:00
status = nbtd_add_socket ( nbtsrv , address , address , bcast , netmask ) ;
2005-01-30 10:24:36 +00:00
NT_STATUS_NOT_OK_RETURN ( status ) ;
}
2005-02-06 08:25:53 +00:00
if ( lp_wins_server_list ( ) ) {
status = nbtd_add_wins_socket ( nbtsrv ) ;
NT_STATUS_NOT_OK_RETURN ( status ) ;
}
2005-01-30 10:24:36 +00:00
talloc_free ( tmp_ctx ) ;
return NT_STATUS_OK ;
}
2005-02-06 08:25:53 +00:00
/*
form a list of addresses that we should use in name query replies
2005-02-07 11:49:55 +00:00
we always place the IP in the given interface first
2005-02-06 08:25:53 +00:00
*/
2005-02-07 11:49:55 +00:00
const char * * nbtd_address_list ( struct nbtd_interface * iface , TALLOC_CTX * mem_ctx )
2005-02-06 08:25:53 +00:00
{
2005-02-07 11:49:55 +00:00
struct nbtd_server * nbtsrv = iface - > nbtsrv ;
2005-02-06 08:25:53 +00:00
const char * * ret = NULL ;
2005-02-07 11:49:55 +00:00
struct nbtd_interface * iface2 ;
2005-12-29 16:58:35 +00:00
BOOL is_loopback = False ;
2005-02-07 11:49:55 +00:00
2005-02-08 01:09:21 +00:00
if ( iface - > ip_address ) {
2005-12-29 16:58:35 +00:00
is_loopback = iface_same_net ( iface - > ip_address , " 127.0.0.1 " , " 255.0.0.0 " ) ;
2005-12-28 04:55:53 +00:00
ret = str_list_add ( ret , iface - > ip_address ) ;
2005-02-08 01:09:21 +00:00
}
2005-02-07 11:49:55 +00:00
for ( iface2 = nbtsrv - > interfaces ; iface2 ; iface2 = iface2 - > next ) {
2005-12-29 16:58:35 +00:00
if ( iface2 = = iface ) continue ;
if ( ! iface2 - > ip_address ) continue ;
if ( ! is_loopback ) {
if ( iface_same_net ( iface2 - > ip_address , " 127.0.0.1 " , " 255.0.0.0 " ) ) {
continue ;
}
2005-02-07 11:49:55 +00:00
}
2005-02-06 08:25:53 +00:00
2005-12-28 04:55:53 +00:00
ret = str_list_add ( ret , iface2 - > ip_address ) ;
2005-02-06 08:25:53 +00:00
}
2005-12-22 22:03:25 +00:00
2005-12-28 04:55:53 +00:00
talloc_steal ( mem_ctx , ret ) ;
2005-12-22 22:03:25 +00:00
2005-12-28 04:55:53 +00:00
return ret ;
2005-02-06 08:25:53 +00:00
}
2005-11-14 01:50:55 +00:00
/*
find the interface to use for sending a outgoing request
*/
2006-03-25 09:24:53 +00:00
struct nbtd_interface * nbtd_find_request_iface ( struct nbtd_server * nbtd_server ,
const char * address , BOOL allow_bcast_iface )
2005-11-14 01:50:55 +00:00
{
2006-03-25 09:24:53 +00:00
struct nbtd_interface * cur ;
2005-11-14 01:50:55 +00:00
/* try to find a exact match */
2006-03-25 09:24:53 +00:00
for ( cur = nbtd_server - > interfaces ; cur ; cur = cur - > next ) {
if ( iface_same_net ( address , cur - > ip_address , cur - > netmask ) ) {
return cur ;
2005-11-14 01:50:55 +00:00
}
}
/* no exact match, if we have the broadcast interface, use that */
2006-03-25 09:24:53 +00:00
if ( allow_bcast_iface & & nbtd_server - > bcast_interface ) {
2005-11-14 01:50:55 +00:00
return nbtd_server - > bcast_interface ;
}
/* fallback to first interface */
return nbtd_server - > interfaces ;
}
2006-03-25 09:24:53 +00:00
/*
* find the interface to use for sending a outgoing reply
*/
struct nbtd_interface * nbtd_find_reply_iface ( struct nbtd_interface * iface ,
const char * address , BOOL allow_bcast_iface )
{
struct nbtd_server * nbtd_server = iface - > nbtsrv ;
/* first try to use the given interfacel when it's not the broadcast one */
if ( iface ! = nbtd_server - > bcast_interface ) {
return iface ;
}
return nbtd_find_request_iface ( nbtd_server , address , allow_bcast_iface ) ;
}