2005-01-26 03:19:42 +00:00
/*
Unix SMB / CIFS implementation .
2005-01-28 11:26:51 +00:00
2005-01-26 03:19:42 +00:00
NBT client - used to lookup netbios names
2005-01-28 11:26:51 +00:00
Copyright ( C ) Andrew Tridgell 1994 - 2005
2005-01-26 03:19:42 +00:00
Copyright ( C ) Jelmer Vernooij 2003 ( Conversion to popt )
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"
2005-01-28 11:26:51 +00:00
# include "dynconfig.h"
# include "libcli/nbt/libnbt.h"
# include "lib/cmdline/popt_common.h"
# include "system/iconv.h"
/* command line options */
static struct {
const char * broadcast_address ;
const char * unicast_address ;
BOOL find_master ;
BOOL wins_lookup ;
BOOL node_status ;
BOOL root_port ;
BOOL lookup_by_ip ;
} options ;
/*
clean any binary from a node name
*/
static const char * clean_name ( TALLOC_CTX * mem_ctx , const char * name )
{
char * ret = talloc_strdup ( mem_ctx , name ) ;
int i ;
for ( i = 0 ; ret [ i ] ; i + + ) {
if ( ! isprint ( ret [ i ] ) ) ret [ i ] = ' . ' ;
}
return ret ;
}
2005-01-26 03:19:42 +00:00
2005-01-28 11:26:51 +00:00
/*
turn a node status flags field into a string
*/
static char * node_status_flags ( TALLOC_CTX * mem_ctx , uint16_t flags )
2005-01-26 03:19:42 +00:00
{
2005-01-28 11:26:51 +00:00
char * ret ;
const char * group = " " ;
const char * type = " B " ;
2005-01-26 03:19:42 +00:00
2005-01-28 11:26:51 +00:00
if ( flags & NBT_NM_GROUP ) {
group = " <GROUP> " ;
}
2005-01-26 03:19:42 +00:00
2005-01-28 11:26:51 +00:00
switch ( flags & NBT_NM_OWNER_TYPE ) {
case NBT_NODE_B :
type = " B " ;
break ;
case NBT_NODE_P :
type = " P " ;
break ;
case NBT_NODE_M :
type = " M " ;
break ;
case NBT_NODE_H :
type = " H " ;
break ;
}
2005-01-26 03:19:42 +00:00
2005-01-28 11:26:51 +00:00
ret = talloc_asprintf ( mem_ctx , " %s %s " , group , type ) ;
2005-01-26 03:19:42 +00:00
2005-01-28 11:26:51 +00:00
if ( flags & NBT_NM_DEREGISTER ) {
ret = talloc_asprintf_append ( ret , " <DEREGISTERING> " ) ;
}
if ( flags & NBT_NM_CONFLICT ) {
ret = talloc_asprintf_append ( ret , " <CONFLICT> " ) ;
}
if ( flags & NBT_NM_ACTIVE ) {
ret = talloc_asprintf_append ( ret , " <ACTIVE> " ) ;
}
if ( flags & NBT_NM_PERMANENT ) {
ret = talloc_asprintf_append ( ret , " <PERMANENT> " ) ;
}
2005-01-26 03:19:42 +00:00
return ret ;
}
2005-01-28 11:26:51 +00:00
/* do a single node status */
static void do_node_status ( struct nbt_name_socket * nbtsock ,
const char * addr )
2005-01-26 03:19:42 +00:00
{
2005-01-28 11:26:51 +00:00
struct nbt_name_status io ;
NTSTATUS status ;
io . in . name . name = " * " ;
io . in . name . type = 0 ;
io . in . name . scope = NULL ;
io . in . dest_addr = addr ;
2005-02-06 08:27:12 +00:00
io . in . timeout = 1 ;
io . in . retries = 2 ;
2005-01-28 11:26:51 +00:00
status = nbt_name_status ( nbtsock , nbtsock , & io ) ;
if ( NT_STATUS_IS_OK ( status ) ) {
int i ;
printf ( " Node status reply from %s \n " ,
io . out . reply_from ) ;
for ( i = 0 ; i < io . out . status . num_names ; i + + ) {
d_printf ( " \t %-16s <%02x> %s \n " ,
clean_name ( nbtsock , io . out . status . names [ i ] . name ) ,
io . out . status . names [ i ] . type ,
node_status_flags ( nbtsock , io . out . status . names [ i ] . nb_flags ) ) ;
}
printf ( " \n \t MAC Address = %02X-%02X-%02X-%02X-%02X-%02X \n " ,
io . out . status . statistics . unit_id [ 0 ] ,
io . out . status . statistics . unit_id [ 1 ] ,
io . out . status . statistics . unit_id [ 2 ] ,
io . out . status . statistics . unit_id [ 3 ] ,
io . out . status . statistics . unit_id [ 4 ] ,
io . out . status . statistics . unit_id [ 5 ] ) ;
}
2005-01-26 03:19:42 +00:00
}
2005-01-28 11:26:51 +00:00
/* do a single node query */
static NTSTATUS do_node_query ( struct nbt_name_socket * nbtsock ,
const char * addr ,
const char * node_name ,
enum nbt_name_type node_type ,
BOOL broadcast )
2005-01-26 03:19:42 +00:00
{
2005-01-28 11:26:51 +00:00
struct nbt_name_query io ;
NTSTATUS status ;
2005-01-31 01:57:58 +00:00
int i ;
2005-01-28 11:26:51 +00:00
io . in . name . name = node_name ;
io . in . name . type = node_type ;
io . in . name . scope = NULL ;
io . in . dest_addr = addr ;
io . in . broadcast = broadcast ;
io . in . wins_lookup = options . wins_lookup ;
2005-02-06 08:27:12 +00:00
io . in . timeout = 1 ;
io . in . retries = 2 ;
2005-01-28 11:26:51 +00:00
status = nbt_name_query ( nbtsock , nbtsock , & io ) ;
2005-01-31 01:57:58 +00:00
NT_STATUS_NOT_OK_RETURN ( status ) ;
for ( i = 0 ; i < io . out . num_addrs ; i + + ) {
2005-01-28 11:26:51 +00:00
printf ( " %s %s<%02x> \n " ,
2005-01-31 01:57:58 +00:00
io . out . reply_addrs [ i ] ,
2005-01-28 11:26:51 +00:00
io . out . name . name ,
io . out . name . type ) ;
2005-01-31 01:57:58 +00:00
}
if ( options . node_status & & io . out . num_addrs > 0 ) {
do_node_status ( nbtsock , io . out . reply_addrs [ 0 ] ) ;
2005-01-26 03:19:42 +00:00
}
2005-01-28 11:26:51 +00:00
return status ;
2005-01-26 03:19:42 +00:00
}
2005-01-28 11:26:51 +00:00
static void process_one ( const char * name )
2005-01-26 03:19:42 +00:00
{
2005-01-28 11:26:51 +00:00
TALLOC_CTX * tmp_ctx = talloc_new ( NULL ) ;
enum nbt_name_type node_type = NBT_NAME_CLIENT ;
char * node_name , * p ;
struct nbt_name_socket * nbtsock ;
if ( options . find_master ) {
node_type = NBT_NAME_MASTER ;
if ( * name = = ' - ' ) {
name = " \01 \02 __MSBROWSE__ \02 " ;
node_type = NBT_NAME_MS ;
2005-01-26 03:19:42 +00:00
}
}
2005-01-28 11:26:51 +00:00
p = strchr ( name , ' # ' ) ;
if ( p ) {
node_name = talloc_strndup ( tmp_ctx , name , PTR_DIFF ( p , name ) ) ;
node_type = ( enum nbt_name_type ) strtol ( p + 1 , NULL , 16 ) ;
} else {
node_name = talloc_strdup ( tmp_ctx , name ) ;
}
2005-01-26 03:19:42 +00:00
2005-01-28 11:26:51 +00:00
nbtsock = nbt_name_socket_init ( tmp_ctx , NULL ) ;
2005-01-26 03:19:42 +00:00
2005-01-28 11:26:51 +00:00
if ( options . root_port ) {
NTSTATUS status ;
status = socket_listen ( nbtsock - > sock , " 0.0.0.0 " , NBT_NAME_SERVICE_PORT , 0 , 0 ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " Failed to bind to local port 137 - %s \n " , nt_errstr ( status ) ) ;
return ;
2005-01-26 03:19:42 +00:00
}
}
2005-01-28 11:26:51 +00:00
if ( options . lookup_by_ip ) {
do_node_status ( nbtsock , name ) ;
talloc_free ( tmp_ctx ) ;
return ;
2005-01-26 03:19:42 +00:00
}
2005-01-28 11:26:51 +00:00
if ( options . broadcast_address ) {
do_node_query ( nbtsock , options . broadcast_address , node_name , node_type , True ) ;
} else if ( options . unicast_address ) {
do_node_query ( nbtsock , options . unicast_address , node_name , node_type , False ) ;
} else {
int i , num_interfaces = iface_count ( ) ;
for ( i = 0 ; i < num_interfaces ; i + + ) {
const char * bcast = sys_inet_ntoa ( * iface_n_bcast ( i ) ) ;
NTSTATUS status = do_node_query ( nbtsock , bcast , node_name , node_type , True ) ;
if ( NT_STATUS_IS_OK ( status ) ) break ;
}
}
2005-01-26 03:19:42 +00:00
2005-01-28 11:26:51 +00:00
talloc_free ( tmp_ctx ) ;
}
2005-01-26 03:19:42 +00:00
2005-01-28 11:26:51 +00:00
/*
2005-01-26 03:19:42 +00:00
main program
2005-01-28 11:26:51 +00:00
*/
2005-01-26 03:19:42 +00:00
int main ( int argc , char * argv [ ] )
{
2005-01-28 11:26:51 +00:00
poptContext pc ;
struct poptOption long_options [ ] = {
POPT_AUTOHELP
{ " broadcast " , ' B ' , POPT_ARG_STRING , & options . broadcast_address ,
' B ' , " Specify address to use for broadcasts " , " BROADCAST-ADDRESS " } ,
{ " unicast " , ' U ' , POPT_ARG_STRING , & options . unicast_address ,
' U ' , " Specify address to use for unicast " } ,
{ " master-browser " , ' M ' , POPT_ARG_VAL , & options . find_master ,
True , " Search for a master browser " } ,
{ " wins " , ' W ' , POPT_ARG_VAL , & options . wins_lookup , True , " Do a WINS lookup " } ,
{ " status " , ' S ' , POPT_ARG_VAL , & options . node_status ,
True , " Lookup node status as well " } ,
{ " root-port " , ' r ' , POPT_ARG_VAL , & options . root_port ,
True , " Use root port 137 (Win95 only replies to this) " } ,
{ " lookup-by-ip " , ' A ' , POPT_ARG_VAL , & options . lookup_by_ip ,
True , " Do a node status on <name> as an IP Address " } ,
POPT_COMMON_SAMBA
{ 0 , 0 , 0 , 0 }
} ;
setup_logging ( argv [ 0 ] , True ) ;
pc = poptGetContext ( " nmblookup " , argc , ( const char * * ) argv , long_options ,
POPT_CONTEXT_KEEP_FIRST ) ;
2005-01-26 03:19:42 +00:00
2005-01-28 11:26:51 +00:00
poptSetOtherOptionHelp ( pc , " <NODE> ... " ) ;
while ( ( poptGetNextOpt ( pc ) ! = - 1 ) ) /* noop */ ;
/* swallow argv[0] */
poptGetArg ( pc ) ;
if ( ! poptPeekArg ( pc ) ) {
poptPrintUsage ( pc , stderr , 0 ) ;
exit ( 1 ) ;
}
lp_load ( dyn_CONFIGFILE , True , False , False ) ;
load_interfaces ( ) ;
while ( poptPeekArg ( pc ) ) {
const char * name = poptGetArg ( pc ) ;
process_one ( name ) ;
}
poptFreeContext ( pc ) ;
return 0 ;
2005-01-26 03:19:42 +00:00
}