1999-12-17 09:11:25 +03:00
/*
Unix SMB / Netbios implementation .
Version 2.0
a WINS nsswitch module
Copyright ( C ) Andrew Tridgell 1999
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 .
*/
# define NO_SYSLOG
# include "includes.h"
# include <nss.h>
# ifndef INADDRSZ
# define INADDRSZ 4
# endif
2001-05-16 01:53:19 +04:00
/* Use our own create socket code so we don't recurse.... */
static int wins_lookup_open_socket_in ( void )
{
struct sockaddr_in sock ;
int val = 1 ;
int res ;
memset ( ( char * ) & sock , ' \0 ' , sizeof ( sock ) ) ;
# ifdef HAVE_SOCK_SIN_LEN
sock . sin_len = sizeof ( sock ) ;
# endif
sock . sin_port = 0 ;
sock . sin_family = AF_INET ;
sock . sin_addr . s_addr = interpret_addr ( " 0.0.0.0 " ) ;
res = socket ( AF_INET , SOCK_DGRAM , 0 ) ;
if ( res = = - 1 )
return - 1 ;
setsockopt ( res , SOL_SOCKET , SO_REUSEADDR , ( char * ) & val , sizeof ( val ) ) ;
# ifdef SO_REUSEPORT
setsockopt ( res , SOL_SOCKET , SO_REUSEPORT , ( char * ) & val , sizeof ( val ) ) ;
# endif /* SO_REUSEPORT */
/* now we've got a socket - we need to bind it */
if ( bind ( res , ( struct sockaddr * ) & sock , sizeof ( sock ) ) < 0 )
return ( - 1 ) ;
return res ;
}
1999-12-17 09:11:25 +03:00
struct in_addr * lookup_backend ( const char * name , int * count )
{
int fd ;
static int initialised ;
struct in_addr * ret ;
2000-10-11 19:54:17 +04:00
struct in_addr p ;
1999-12-17 09:11:25 +03:00
int j ;
if ( ! initialised ) {
initialised = 1 ;
DEBUGLEVEL = 0 ;
setup_logging ( " nss_wins " , True ) ;
2001-11-19 05:49:53 +03:00
lp_load ( dyn_CONFIGFILE , True , False , False ) ;
1999-12-17 09:11:25 +03:00
load_interfaces ( ) ;
}
* count = 0 ;
2001-05-16 01:53:19 +04:00
fd = wins_lookup_open_socket_in ( ) ;
if ( fd = = - 1 )
return NULL ;
1999-12-17 09:11:25 +03:00
set_socket_options ( fd , " SO_BROADCAST " ) ;
2000-10-10 01:44:00 +04:00
/* The next four lines commented out by JHT
and replaced with the four lines following */
2001-11-26 06:11:44 +03:00
/* if( !is_zero_ip( wins_ip ) ) {
2000-10-10 01:44:00 +04:00
* ret = name_query ( fd , name , 0x20 , False , True , wins_src_ip ( ) , count ) ;
* goto out ;
* }
*/
2000-10-11 19:54:17 +04:00
p = wins_srv_ip ( ) ;
2001-11-26 06:11:44 +03:00
if ( ! is_zero_ip ( p ) ) {
2000-10-11 19:54:17 +04:00
ret = name_query ( fd , name , 0x20 , False , True , p , count ) ;
1999-12-17 09:11:25 +03:00
goto out ;
}
if ( lp_wins_support ( ) ) {
/* we are our own WINS server */
2000-01-07 11:46:33 +03:00
ret = name_query ( fd , name , 0x20 , False , True , * interpret_addr2 ( " 127.0.0.1 " ) , count ) ;
1999-12-17 09:11:25 +03:00
goto out ;
}
/* uggh, we have to broadcast to each interface in turn */
for ( j = iface_count ( ) - 1 ;
j > = 0 ;
j - - ) {
struct in_addr * bcast = iface_n_bcast ( j ) ;
2000-01-07 11:46:33 +03:00
ret = name_query ( fd , name , 0x20 , True , True , * bcast , count ) ;
1999-12-17 09:11:25 +03:00
if ( ret ) break ;
}
out :
2001-05-16 01:53:19 +04:00
1999-12-17 09:11:25 +03:00
close ( fd ) ;
return ret ;
}
/****************************************************************************
gethostbyname ( ) - we ignore any domain portion of the name and only
handle names that are at most 15 characters long
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
enum nss_status
_nss_wins_gethostbyname_r ( const char * name , struct hostent * he ,
char * buffer , size_t buflen , int * errnop ,
int * h_errnop )
{
char * * host_addresses ;
struct in_addr * ip_list ;
int i , count ;
2001-05-16 01:53:19 +04:00
size_t namelen = strlen ( name ) + 1 ;
1999-12-17 09:11:25 +03:00
2001-05-16 02:01:48 +04:00
memset ( he , ' \0 ' , sizeof ( * he ) ) ;
1999-12-17 09:11:25 +03:00
ip_list = lookup_backend ( name , & count ) ;
if ( ! ip_list ) {
return NSS_STATUS_NOTFOUND ;
}
2001-05-16 01:53:19 +04:00
if ( buflen < namelen + ( 2 * count + 1 ) * INADDRSZ ) {
1999-12-17 09:11:25 +03:00
/* no ENOMEM error type?! */
return NSS_STATUS_NOTFOUND ;
}
host_addresses = ( char * * ) buffer ;
he - > h_addr_list = host_addresses ;
host_addresses [ count ] = NULL ;
buffer + = ( count + 1 ) * INADDRSZ ;
buflen + = ( count + 1 ) * INADDRSZ ;
he - > h_addrtype = AF_INET ;
he - > h_length = INADDRSZ ;
for ( i = 0 ; i < count ; i + + ) {
memcpy ( buffer , & ip_list [ i ] . s_addr , INADDRSZ ) ;
* host_addresses = buffer ;
buffer + = INADDRSZ ;
buflen - = INADDRSZ ;
host_addresses + + ;
}
2001-09-17 08:52:45 +04:00
SAFE_FREE ( ip_list ) ;
2001-05-16 01:53:19 +04:00
memcpy ( buffer , name , namelen ) ;
he - > h_name = buffer ;
1999-12-17 09:11:25 +03:00
return NSS_STATUS_SUCCESS ;
}