2007-10-02 19:50:53 +00:00
/*
2002-01-30 06:08:46 +00:00
Unix SMB / CIFS implementation .
1998-11-09 16:40:38 +00:00
Samba utility functions
Copyright ( C ) Andrew Tridgell 1992 - 1998
2002-01-13 11:13:54 +00:00
Copyright ( C ) Tim Potter 2000 - 2001
2007-10-03 20:43:55 +00:00
Copyright ( C ) Jeremy Allison 1992 - 2007
2007-10-02 19:50:53 +00:00
1998-11-09 16:40:38 +00:00
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-09 19:25:36 +00:00
the Free Software Foundation ; either version 3 of the License , or
1998-11-09 16:40:38 +00:00
( at your option ) any later version .
2007-10-02 19:50:53 +00:00
1998-11-09 16:40:38 +00:00
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 .
2007-10-02 19:50:53 +00:00
1998-11-09 16:40:38 +00:00
You should have received a copy of the GNU General Public License
2007-07-10 00:52:41 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
1998-11-09 16:40:38 +00:00
*/
# include "includes.h"
2011-02-25 23:20:06 +01:00
# include "system/filesys.h"
2014-07-17 12:58:34 +02:00
# include "../lib/util/memcache.h"
2010-08-26 09:58:09 +02:00
# include "../lib/async_req/async_sock.h"
2010-10-01 10:08:15 +02:00
# include "../lib/util/select.h"
2011-05-02 12:20:21 +10:00
# include "lib/socket/interfaces.h"
2011-04-28 17:38:09 +02:00
# include "../lib/util/tevent_unix.h"
# include "../lib/util/tevent_ntstatus.h"
2011-06-08 14:50:20 +02:00
# include "../lib/tsocket/tsocket.h"
2014-11-19 13:33:06 +00:00
# include "lib/sys_rw.h"
2014-11-19 14:25:56 +00:00
# include "lib/sys_rw_data.h"
1998-11-09 16:40:38 +00:00
2007-11-03 23:20:10 -07:00
const char * client_addr ( int fd , char * addr , size_t addrlen )
2005-06-13 22:26:08 +00:00
{
2007-11-03 23:20:10 -07:00
return get_peer_addr ( fd , addr , addrlen ) ;
2005-06-13 22:26:08 +00:00
}
2007-11-03 23:20:10 -07:00
#if 0
/* Not currently used. JRA. */
int client_socket_port ( int fd )
2005-06-13 22:26:08 +00:00
{
2007-11-03 23:20:10 -07:00
return get_socket_port ( fd ) ;
2005-06-13 22:26:08 +00:00
}
2007-11-03 23:20:10 -07:00
# endif
2005-06-13 22:26:08 +00:00
1998-11-09 16:40:38 +00:00
/****************************************************************************
1999-12-13 13:27:58 +00:00
Determine if a file descriptor is in fact a socket .
1998-11-09 16:40:38 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 13:27:58 +00:00
2007-10-10 18:25:16 -07:00
bool is_a_socket ( int fd )
1998-11-09 16:40:38 +00:00
{
2004-05-05 03:03:38 +00:00
int v ;
socklen_t l ;
2002-07-15 10:35:28 +00:00
l = sizeof ( int ) ;
return ( getsockopt ( fd , SOL_SOCKET , SO_TYPE , ( char * ) & v , & l ) = = 0 ) ;
1998-11-09 16:40:38 +00:00
}
/****************************************************************************
1999-12-13 13:27:58 +00:00
Read from a socket .
1998-11-09 16:40:38 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-03 20:43:55 +00:00
ssize_t read_udp_v4_socket ( int fd ,
char * buf ,
size_t len ,
struct sockaddr_storage * psa )
1998-11-09 16:40:38 +00:00
{
2001-06-27 17:26:29 +00:00
ssize_t ret ;
2007-10-03 20:43:55 +00:00
socklen_t socklen = sizeof ( * psa ) ;
struct sockaddr_in * si = ( struct sockaddr_in * ) psa ;
memset ( ( char * ) psa , ' \0 ' , socklen ) ;
2001-06-27 17:26:29 +00:00
2007-10-02 19:50:53 +00:00
ret = ( ssize_t ) sys_recvfrom ( fd , buf , len , 0 ,
2007-10-03 20:43:55 +00:00
( struct sockaddr * ) psa , & socklen ) ;
2001-06-27 17:26:29 +00:00
if ( ret < = 0 ) {
2006-05-18 19:49:44 +00:00
/* Don't print a low debug error for a non-blocking socket. */
if ( errno = = EAGAIN ) {
2007-10-03 20:43:55 +00:00
DEBUG ( 10 , ( " read_udp_v4_socket: returned EAGAIN \n " ) ) ;
2006-05-18 19:49:44 +00:00
} else {
2007-10-03 20:43:55 +00:00
DEBUG ( 2 , ( " read_udp_v4_socket: failed. errno=%s \n " ,
strerror ( errno ) ) ) ;
2006-05-18 19:49:44 +00:00
}
2007-10-03 20:43:55 +00:00
return 0 ;
2001-06-27 17:26:29 +00:00
}
2007-10-03 20:43:55 +00:00
if ( psa - > ss_family ! = AF_INET ) {
2007-10-05 01:11:33 +00:00
DEBUG ( 2 , ( " read_udp_v4_socket: invalid address family %d "
2007-10-03 20:43:55 +00:00
" (not IPv4) \n " , ( int ) psa - > ss_family ) ) ;
return 0 ;
}
2001-06-27 17:26:29 +00:00
2007-10-05 01:11:33 +00:00
DEBUG ( 10 , ( " read_udp_v4_socket: ip %s port %d read: %lu \n " ,
2007-10-03 20:43:55 +00:00
inet_ntoa ( si - > sin_addr ) ,
si - > sin_port ,
( unsigned long ) ret ) ) ;
2001-06-27 17:26:29 +00:00
2007-10-03 20:43:55 +00:00
return ret ;
1998-11-09 16:40:38 +00:00
}
/****************************************************************************
2009-09-06 21:38:50 -07:00
Read data from a file descriptor with a timout in msec .
1999-12-13 13:27:58 +00:00
mincount = if timeout , minimum to read before returning
maxcount = number to be read .
time_out = timeout in milliseconds
2009-09-06 21:38:50 -07:00
NB . This can be called with a non - socket fd , don ' t change
sys_read ( ) to sys_recv ( ) or other socket call .
1998-11-09 16:40:38 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-09-06 21:38:50 -07:00
NTSTATUS read_fd_with_timeout ( int fd , char * buf ,
2008-01-25 23:43:50 +01:00
size_t mincnt , size_t maxcnt ,
unsigned int time_out ,
size_t * size_ret )
1998-11-09 16:40:38 +00:00
{
2011-02-08 17:57:12 +01:00
int pollrtn ;
2002-01-13 12:33:42 +00:00
ssize_t readret ;
size_t nread = 0 ;
2007-10-02 19:50:53 +00:00
2002-01-13 12:33:42 +00:00
/* just checking .... */
if ( maxcnt < = 0 )
2008-01-24 19:17:14 +01:00
return NT_STATUS_OK ;
2007-10-02 19:50:53 +00:00
2002-01-13 12:33:42 +00:00
/* Blocking read */
2005-06-13 22:26:08 +00:00
if ( time_out = = 0 ) {
if ( mincnt = = 0 ) {
mincnt = maxcnt ;
}
2007-10-02 19:50:53 +00:00
2002-01-13 12:33:42 +00:00
while ( nread < mincnt ) {
2009-09-06 21:38:50 -07:00
readret = sys_read ( fd , buf + nread , maxcnt - nread ) ;
2007-10-02 19:50:53 +00:00
2002-01-13 12:33:42 +00:00
if ( readret = = 0 ) {
2009-09-06 21:38:50 -07:00
DEBUG ( 5 , ( " read_fd_with_timeout: "
2007-10-02 19:50:53 +00:00
" blocking read. EOF from client. \n " ) ) ;
2008-01-24 19:17:14 +01:00
return NT_STATUS_END_OF_FILE ;
2002-01-13 12:33:42 +00:00
}
2007-10-02 19:50:53 +00:00
2002-01-13 12:33:42 +00:00
if ( readret = = - 1 ) {
2010-08-18 11:17:52 +02:00
return map_nt_error_from_unix ( errno ) ;
2002-01-13 12:33:42 +00:00
}
nread + = readret ;
}
2008-01-24 19:17:14 +01:00
goto done ;
2002-01-13 12:33:42 +00:00
}
2007-10-02 19:50:53 +00:00
2002-01-13 12:33:42 +00:00
/* Most difficult - timeout read */
2007-10-02 19:50:53 +00:00
/* If this is ever called on a disk file and
2002-01-13 12:33:42 +00:00
mincnt is greater then the filesize then
2007-10-02 19:50:53 +00:00
system performance will suffer severely as
2002-01-13 12:33:42 +00:00
select always returns true on disk files */
2007-10-02 19:50:53 +00:00
for ( nread = 0 ; nread < mincnt ; ) {
2011-02-08 17:57:12 +01:00
int revents ;
2007-10-02 19:50:53 +00:00
2011-02-08 17:57:12 +01:00
pollrtn = poll_intr_one_fd ( fd , POLLIN | POLLHUP , time_out ,
& revents ) ;
2007-10-02 19:50:53 +00:00
2002-01-13 12:33:42 +00:00
/* Check if error */
2011-02-08 17:57:12 +01:00
if ( pollrtn = = - 1 ) {
2010-08-18 11:17:52 +02:00
return map_nt_error_from_unix ( errno ) ;
2002-01-13 12:33:42 +00:00
}
2007-10-02 19:50:53 +00:00
2002-01-13 12:33:42 +00:00
/* Did we timeout ? */
2011-02-08 17:57:12 +01:00
if ( ( pollrtn = = 0 ) | |
( ( revents & ( POLLIN | POLLHUP | POLLERR ) ) = = 0 ) ) {
2009-09-06 21:38:50 -07:00
DEBUG ( 10 , ( " read_fd_with_timeout: timeout read. "
2007-10-02 19:50:53 +00:00
" select timed out. \n " ) ) ;
2008-01-24 19:17:14 +01:00
return NT_STATUS_IO_TIMEOUT ;
2002-01-13 12:33:42 +00:00
}
2007-10-02 19:50:53 +00:00
2009-09-06 21:38:50 -07:00
readret = sys_read ( fd , buf + nread , maxcnt - nread ) ;
2007-10-02 19:50:53 +00:00
2002-01-13 12:33:42 +00:00
if ( readret = = 0 ) {
/* we got EOF on the file descriptor */
2009-09-06 21:38:50 -07:00
DEBUG ( 5 , ( " read_fd_with_timeout: timeout read. "
2007-10-02 19:50:53 +00:00
" EOF from client. \n " ) ) ;
2008-01-24 19:17:14 +01:00
return NT_STATUS_END_OF_FILE ;
2002-01-13 12:33:42 +00:00
}
2007-10-02 19:50:53 +00:00
2002-01-13 12:33:42 +00:00
if ( readret = = - 1 ) {
2008-01-24 19:17:14 +01:00
return map_nt_error_from_unix ( errno ) ;
2002-01-13 12:33:42 +00:00
}
2007-10-02 19:50:53 +00:00
2002-01-13 12:33:42 +00:00
nread + = readret ;
}
2007-10-02 19:50:53 +00:00
2008-01-24 19:17:14 +01:00
done :
2002-01-13 12:33:42 +00:00
/* Return the number we got */
2008-01-24 19:17:14 +01:00
if ( size_ret ) {
* size_ret = nread ;
}
return NT_STATUS_OK ;
}
2008-01-25 23:41:48 +01:00
/****************************************************************************
2009-09-06 21:38:50 -07:00
Read data from an fd , reading exactly N bytes .
NB . This can be called with a non - socket fd , don ' t add dependencies
on socket calls .
2008-01-25 23:41:48 +01:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2014-11-19 14:06:49 +00:00
NTSTATUS read_data_ntstatus ( int fd , char * buffer , size_t N )
2008-01-24 19:17:14 +01:00
{
2010-08-15 15:30:21 +02:00
return read_fd_with_timeout ( fd , buffer , N , N , 0 , NULL ) ;
1998-11-09 16:40:38 +00:00
}
2002-07-15 10:35:28 +00:00
/****************************************************************************
2003-07-16 21:06:21 +00:00
Send a keepalive packet ( rfc1002 ) .
2002-07-15 10:35:28 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 13:27:58 +00:00
2007-10-10 18:25:16 -07:00
bool send_keepalive ( int client )
2002-07-15 10:35:28 +00:00
{
unsigned char buf [ 4 ] ;
2011-07-12 08:53:30 +02:00
buf [ 0 ] = NBSSkeepalive ;
2002-07-15 10:35:28 +00:00
buf [ 1 ] = buf [ 2 ] = buf [ 3 ] = 0 ;
2005-06-13 22:26:08 +00:00
return ( write_data ( client , ( char * ) buf , 4 ) = = 4 ) ;
1999-12-13 13:27:58 +00:00
}
1998-11-09 16:40:38 +00:00
/****************************************************************************
2003-07-16 21:06:21 +00:00
Read 4 bytes of a smb packet and return the smb length of the packet .
Store the result in the buffer .
This version of the function will return a length of zero on receiving
a keepalive packet .
Timeout is in milliseconds .
1998-11-09 16:40:38 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 13:27:58 +00:00
2008-01-25 21:24:48 +01:00
NTSTATUS read_smb_length_return_keepalive ( int fd , char * inbuf ,
unsigned int timeout ,
size_t * len )
1998-11-09 16:40:38 +00:00
{
2002-07-15 10:35:28 +00:00
int msg_type ;
2008-01-25 21:02:52 +01:00
NTSTATUS status ;
2009-09-06 21:38:50 -07:00
status = read_fd_with_timeout ( fd , inbuf , 4 , 4 , timeout , NULL ) ;
2008-01-25 21:02:52 +01:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2008-01-25 21:24:48 +01:00
return status ;
2008-01-25 09:21:44 +01:00
}
1998-11-09 16:40:38 +00:00
2008-01-25 21:24:48 +01:00
* len = smb_len ( inbuf ) ;
2008-01-25 09:21:44 +01:00
msg_type = CVAL ( inbuf , 0 ) ;
1998-11-09 16:40:38 +00:00
2011-07-12 08:53:30 +02:00
if ( msg_type = = NBSSkeepalive ) {
2008-01-25 09:21:44 +01:00
DEBUG ( 5 , ( " Got keepalive packet \n " ) ) ;
2002-07-15 10:35:28 +00:00
}
1998-11-09 16:40:38 +00:00
2008-02-05 22:17:20 +01:00
DEBUG ( 10 , ( " got smb length of %lu \n " , ( unsigned long ) ( * len ) ) ) ;
1998-11-09 16:40:38 +00:00
2008-01-25 21:24:48 +01:00
return NT_STATUS_OK ;
1998-11-09 16:40:38 +00:00
}
/****************************************************************************
2008-05-28 09:31:42 -07:00
Read an smb from a fd .
2007-10-02 19:50:53 +00:00
The timeout is in milliseconds .
2003-07-16 21:06:21 +00:00
This function will return on receipt of a session keepalive packet .
2007-05-16 00:07:38 +00:00
maxlen is the max number of bytes to return , not including the 4 byte
2008-05-28 09:31:42 -07:00
length . If zero it means buflen limit .
2003-10-21 21:19:00 +00:00
Doesn ' t check the MAC on signed packets .
1998-11-09 16:40:38 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 13:27:58 +00:00
2008-05-28 09:31:42 -07:00
NTSTATUS receive_smb_raw ( int fd , char * buffer , size_t buflen , unsigned int timeout ,
2008-01-25 23:54:22 +01:00
size_t maxlen , size_t * p_len )
1998-11-09 16:40:38 +00:00
{
2008-01-25 21:24:48 +01:00
size_t len ;
NTSTATUS status ;
1998-11-09 16:40:38 +00:00
2008-01-25 21:24:48 +01:00
status = read_smb_length_return_keepalive ( fd , buffer , timeout , & len ) ;
2002-02-23 21:03:21 +00:00
2008-01-25 21:24:48 +01:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2010-08-15 15:40:08 +02:00
DEBUG ( 0 , ( " read_fd_with_timeout failed, read "
" error = %s. \n " , nt_errstr ( status ) ) ) ;
2008-01-25 23:54:22 +01:00
return status ;
2001-05-24 19:28:22 +00:00
}
1998-11-09 16:40:38 +00:00
2008-05-28 09:31:42 -07:00
if ( len > buflen ) {
2007-10-02 19:50:53 +00:00
DEBUG ( 0 , ( " Invalid packet length! (%lu bytes). \n " ,
( unsigned long ) len ) ) ;
2008-05-28 09:31:42 -07:00
return NT_STATUS_INVALID_PARAMETER ;
2001-05-24 19:28:22 +00:00
}
if ( len > 0 ) {
2007-05-16 00:07:38 +00:00
if ( maxlen ) {
len = MIN ( len , maxlen ) ;
}
2009-09-06 21:38:50 -07:00
status = read_fd_with_timeout (
2008-01-25 23:41:48 +01:00
fd , buffer + 4 , len , len , timeout , & len ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2010-08-15 15:40:08 +02:00
DEBUG ( 0 , ( " read_fd_with_timeout failed, read error = "
" %s. \n " , nt_errstr ( status ) ) ) ;
2008-01-25 23:54:22 +01:00
return status ;
2001-05-24 19:28:22 +00:00
}
2007-10-02 19:50:53 +00:00
/* not all of samba3 properly checks for packet-termination
* of strings . This ensures that we don ' t run off into
* empty space . */
2005-05-27 17:47:24 +00:00
SSVAL ( buffer + 4 , len , 0 ) ;
2001-05-24 19:28:22 +00:00
}
2008-01-25 23:54:22 +01:00
* p_len = len ;
return NT_STATUS_OK ;
2003-10-21 21:19:00 +00:00
}
1998-11-09 16:40:38 +00:00
/****************************************************************************
2001-08-26 19:11:33 +00:00
Open a socket of the specified type , port , and address for incoming data .
1998-11-09 16:40:38 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 13:27:58 +00:00
2007-10-02 19:50:53 +00:00
int open_socket_in ( int type ,
2007-10-24 14:16:54 -07:00
uint16_t port ,
2007-10-02 19:50:53 +00:00
int dlevel ,
2007-10-24 14:16:54 -07:00
const struct sockaddr_storage * psock ,
bool rebind )
2001-08-26 19:11:33 +00:00
{
2007-10-24 14:16:54 -07:00
struct sockaddr_storage sock ;
2001-08-26 19:11:33 +00:00
int res ;
2007-11-02 14:22:19 -07:00
socklen_t slen = sizeof ( struct sockaddr_in ) ;
1998-11-09 16:40:38 +00:00
2007-10-24 14:16:54 -07:00
sock = * psock ;
1998-11-09 16:40:38 +00:00
2007-10-24 14:16:54 -07:00
# if defined(HAVE_IPV6)
if ( sock . ss_family = = AF_INET6 ) {
( ( struct sockaddr_in6 * ) & sock ) - > sin6_port = htons ( port ) ;
2007-11-02 14:22:19 -07:00
slen = sizeof ( struct sockaddr_in6 ) ;
2007-10-24 14:16:54 -07:00
}
1998-11-09 16:40:38 +00:00
# endif
2007-10-24 14:16:54 -07:00
if ( sock . ss_family = = AF_INET ) {
( ( struct sockaddr_in * ) & sock ) - > sin_port = htons ( port ) ;
}
2001-08-26 19:11:33 +00:00
2007-10-24 14:16:54 -07:00
res = socket ( sock . ss_family , type , 0 ) ;
2001-08-26 19:11:33 +00:00
if ( res = = - 1 ) {
if ( DEBUGLVL ( 0 ) ) {
dbgtext ( " open_socket_in(): socket() call failed: " ) ;
dbgtext ( " %s \n " , strerror ( errno ) ) ;
}
return - 1 ;
}
2001-08-26 04:16:51 +00:00
2001-08-26 19:11:33 +00:00
/* This block sets/clears the SO_REUSEADDR and possibly SO_REUSEPORT. */
{
int val = rebind ? 1 : 0 ;
2007-10-02 19:50:53 +00:00
if ( setsockopt ( res , SOL_SOCKET , SO_REUSEADDR ,
( char * ) & val , sizeof ( val ) ) = = - 1 ) {
2001-08-26 19:11:33 +00:00
if ( DEBUGLVL ( dlevel ) ) {
dbgtext ( " open_socket_in(): setsockopt: " ) ;
2007-10-02 19:50:53 +00:00
dbgtext ( " SO_REUSEADDR = %s " ,
2007-10-10 18:25:16 -07:00
val ? " true " : " false " ) ;
2001-08-26 19:11:33 +00:00
dbgtext ( " on port %d failed " , port ) ;
dbgtext ( " with error = %s \n " , strerror ( errno ) ) ;
}
}
2000-02-16 22:48:19 +00:00
# ifdef SO_REUSEPORT
2007-10-02 19:50:53 +00:00
if ( setsockopt ( res , SOL_SOCKET , SO_REUSEPORT ,
( char * ) & val , sizeof ( val ) ) = = - 1 ) {
2001-08-26 19:11:33 +00:00
if ( DEBUGLVL ( dlevel ) ) {
2001-08-28 01:28:01 +00:00
dbgtext ( " open_socket_in(): setsockopt: " ) ;
2007-10-02 19:50:53 +00:00
dbgtext ( " SO_REUSEPORT = %s " ,
2007-10-24 14:16:54 -07:00
val ? " true " : " false " ) ;
dbgtext ( " on port %d failed " , port ) ;
dbgtext ( " with error = %s \n " , strerror ( errno ) ) ;
2001-08-26 19:11:33 +00:00
}
}
2000-02-16 22:48:19 +00:00
# endif /* SO_REUSEPORT */
2001-08-26 19:11:33 +00:00
}
1998-11-09 16:40:38 +00:00
2011-04-23 11:29:51 +02:00
# ifdef HAVE_IPV6
/*
* As IPV6_V6ONLY is the default on some systems ,
* we better try to be consistent and always use it .
*
* This also avoids using IPv4 via AF_INET6 sockets
* and makes sure % I never resolves to a ' : : ffff : 192.168 .0 .1 '
* string .
*/
if ( sock . ss_family = = AF_INET6 ) {
int val = 1 ;
int ret ;
ret = setsockopt ( res , IPPROTO_IPV6 , IPV6_V6ONLY ,
( const void * ) & val , sizeof ( val ) ) ;
if ( ret = = - 1 ) {
if ( DEBUGLVL ( 0 ) ) {
dbgtext ( " open_socket_in(): IPV6_ONLY failed: " ) ;
dbgtext ( " %s \n " , strerror ( errno ) ) ;
}
close ( res ) ;
return - 1 ;
}
}
# endif
2001-08-26 19:11:33 +00:00
/* now we've got a socket - we need to bind it */
2007-11-02 14:22:19 -07:00
if ( bind ( res , ( struct sockaddr * ) & sock , slen ) = = - 1 ) {
2012-02-14 21:51:35 -06:00
if ( DEBUGLVL ( dlevel ) & & ( port = = NMB_PORT | |
port = = NBT_SMB_PORT | |
port = = TCP_SMB_PORT ) ) {
2007-10-24 14:16:54 -07:00
char addr [ INET6_ADDRSTRLEN ] ;
print_sockaddr ( addr , sizeof ( addr ) ,
& sock ) ;
dbgtext ( " bind failed on port %d " , port ) ;
dbgtext ( " socket_addr = %s. \n " , addr ) ;
dbgtext ( " Error = %s \n " , strerror ( errno ) ) ;
2001-08-26 19:11:33 +00:00
}
2007-10-24 14:16:54 -07:00
close ( res ) ;
2007-10-02 19:50:53 +00:00
return - 1 ;
2001-08-26 19:11:33 +00:00
}
1998-11-09 16:40:38 +00:00
2002-07-15 10:35:28 +00:00
DEBUG ( 10 , ( " bind succeeded on port %d \n " , port ) ) ;
2001-08-26 19:11:33 +00:00
return ( res ) ;
}
1998-11-09 16:40:38 +00:00
2009-01-03 19:50:05 +01:00
struct open_socket_out_state {
int fd ;
2013-02-18 10:00:26 +01:00
struct tevent_context * ev ;
2009-01-03 19:50:05 +01:00
struct sockaddr_storage ss ;
socklen_t salen ;
uint16_t port ;
2011-06-01 11:24:51 +09:30
int wait_usec ;
2009-01-03 19:50:05 +01:00
} ;
2009-02-22 19:49:18 +01:00
static void open_socket_out_connected ( struct tevent_req * subreq ) ;
2009-01-03 19:50:05 +01:00
static int open_socket_out_state_destructor ( struct open_socket_out_state * s )
{
if ( s - > fd ! = - 1 ) {
close ( s - > fd ) ;
}
return 0 ;
}
1998-11-09 16:40:38 +00:00
/****************************************************************************
2003-07-16 21:06:21 +00:00
Create an outgoing socket . timeout is in milliseconds .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 13:27:58 +00:00
2009-02-26 12:34:39 +01:00
struct tevent_req * open_socket_out_send ( TALLOC_CTX * mem_ctx ,
2013-02-18 10:00:26 +01:00
struct tevent_context * ev ,
2009-02-26 12:34:39 +01:00
const struct sockaddr_storage * pss ,
uint16_t port ,
int timeout )
1998-11-09 16:40:38 +00:00
{
2007-10-24 14:16:54 -07:00
char addr [ INET6_ADDRSTRLEN ] ;
2009-02-26 12:34:39 +01:00
struct tevent_req * result , * subreq ;
2009-01-03 19:50:05 +01:00
struct open_socket_out_state * state ;
NTSTATUS status ;
2009-02-26 12:34:39 +01:00
result = tevent_req_create ( mem_ctx , & state ,
struct open_socket_out_state ) ;
if ( result = = NULL ) {
2009-01-03 19:50:05 +01:00
return NULL ;
}
state - > ev = ev ;
state - > ss = * pss ;
state - > port = port ;
2011-06-01 11:24:51 +09:30
state - > wait_usec = 10000 ;
2009-01-03 19:50:05 +01:00
state - > salen = - 1 ;
state - > fd = socket ( state - > ss . ss_family , SOCK_STREAM , 0 ) ;
if ( state - > fd = = - 1 ) {
status = map_nt_error_from_unix ( errno ) ;
goto post_status ;
}
talloc_set_destructor ( state , open_socket_out_state_destructor ) ;
2009-02-26 12:34:39 +01:00
if ( ! tevent_req_set_endtime (
2011-06-01 11:24:51 +09:30
result , ev , timeval_current_ofs_msec ( timeout ) ) ) {
2009-01-03 19:50:05 +01:00
goto fail ;
2003-07-16 21:06:21 +00:00
}
1998-11-09 16:40:38 +00:00
2007-10-24 14:16:54 -07:00
# if defined(HAVE_IPV6)
if ( pss - > ss_family = = AF_INET6 ) {
2009-01-03 19:50:05 +01:00
struct sockaddr_in6 * psa6 ;
psa6 = ( struct sockaddr_in6 * ) & state - > ss ;
2007-10-24 14:16:54 -07:00
psa6 - > sin6_port = htons ( port ) ;
2009-01-03 19:50:05 +01:00
if ( psa6 - > sin6_scope_id = = 0
& & IN6_IS_ADDR_LINKLOCAL ( & psa6 - > sin6_addr ) ) {
setup_linklocal_scope_id (
( struct sockaddr * ) & ( state - > ss ) ) ;
2007-10-25 18:28:36 -07:00
}
2009-01-03 19:50:05 +01:00
state - > salen = sizeof ( struct sockaddr_in6 ) ;
2007-10-24 14:16:54 -07:00
}
# endif
if ( pss - > ss_family = = AF_INET ) {
2009-01-03 19:50:05 +01:00
struct sockaddr_in * psa ;
psa = ( struct sockaddr_in * ) & state - > ss ;
2007-10-24 14:16:54 -07:00
psa - > sin_port = htons ( port ) ;
2009-01-03 19:50:05 +01:00
state - > salen = sizeof ( struct sockaddr_in ) ;
2007-10-24 14:16:54 -07:00
}
1998-11-09 16:40:38 +00:00
2009-06-06 00:47:53 +02:00
if ( pss - > ss_family = = AF_UNIX ) {
state - > salen = sizeof ( struct sockaddr_un ) ;
}
2009-01-03 19:50:05 +01:00
print_sockaddr ( addr , sizeof ( addr ) , & state - > ss ) ;
DEBUG ( 3 , ( " Connecting to %s at port %u \n " , addr , ( unsigned int ) port ) ) ;
subreq = async_connect_send ( state , state - > ev , state - > fd ,
( struct sockaddr * ) & state - > ss ,
2013-05-16 16:11:54 +02:00
state - > salen , NULL , NULL , NULL ) ;
2009-01-03 19:50:05 +01:00
if ( ( subreq = = NULL )
2009-02-22 19:49:18 +01:00
| | ! tevent_req_set_endtime (
subreq , state - > ev ,
2011-06-01 11:24:51 +09:30
timeval_current_ofs ( 0 , state - > wait_usec ) ) ) {
2009-02-26 12:34:39 +01:00
goto fail ;
2009-01-03 19:50:05 +01:00
}
2009-02-28 15:44:30 -05:00
tevent_req_set_callback ( subreq , open_socket_out_connected , result ) ;
2009-01-03 19:50:05 +01:00
return result ;
1998-11-09 16:40:38 +00:00
2009-01-03 19:50:05 +01:00
post_status :
2009-02-26 12:34:39 +01:00
tevent_req_nterror ( result , status ) ;
return tevent_req_post ( result , ev ) ;
2009-01-03 19:50:05 +01:00
fail :
TALLOC_FREE ( result ) ;
return NULL ;
}
2007-10-02 19:50:53 +00:00
2009-02-22 19:49:18 +01:00
static void open_socket_out_connected ( struct tevent_req * subreq )
2009-01-03 19:50:05 +01:00
{
2009-02-28 15:44:30 -05:00
struct tevent_req * req =
tevent_req_callback_data ( subreq , struct tevent_req ) ;
struct open_socket_out_state * state =
tevent_req_data ( req , struct open_socket_out_state ) ;
2009-02-22 19:49:18 +01:00
int ret ;
2009-01-03 19:50:05 +01:00
int sys_errno ;
2003-07-16 21:06:21 +00:00
2009-02-22 19:49:18 +01:00
ret = async_connect_recv ( subreq , & sys_errno ) ;
2009-01-03 19:50:05 +01:00
TALLOC_FREE ( subreq ) ;
2009-02-22 19:49:18 +01:00
if ( ret = = 0 ) {
2009-02-26 12:34:39 +01:00
tevent_req_done ( req ) ;
2009-01-03 19:50:05 +01:00
return ;
}
2009-02-09 09:36:40 -08:00
if (
# ifdef ETIMEDOUT
( sys_errno = = ETIMEDOUT ) | |
# endif
( sys_errno = = EINPROGRESS ) | |
( sys_errno = = EALREADY ) | |
( sys_errno = = EAGAIN ) ) {
2009-01-03 19:50:05 +01:00
/*
* retry
*/
2003-07-16 21:06:21 +00:00
2011-06-01 11:24:51 +09:30
if ( state - > wait_usec < 250000 ) {
state - > wait_usec * = 1.5 ;
2003-07-16 21:06:21 +00:00
}
2009-01-03 19:50:05 +01:00
subreq = async_connect_send ( state , state - > ev , state - > fd ,
( struct sockaddr * ) & state - > ss ,
2013-05-16 16:11:54 +02:00
state - > salen , NULL , NULL , NULL ) ;
2009-02-26 12:34:39 +01:00
if ( tevent_req_nomem ( subreq , req ) ) {
2009-01-03 19:50:05 +01:00
return ;
}
2009-02-22 19:49:18 +01:00
if ( ! tevent_req_set_endtime (
subreq , state - > ev ,
2011-06-01 11:24:51 +09:30
timeval_current_ofs_usec ( state - > wait_usec ) ) ) {
2009-02-26 12:34:39 +01:00
tevent_req_nterror ( req , NT_STATUS_NO_MEMORY ) ;
2009-01-03 19:50:05 +01:00
return ;
}
2009-02-28 15:44:30 -05:00
tevent_req_set_callback ( subreq , open_socket_out_connected , req ) ;
2009-01-03 19:50:05 +01:00
return ;
2003-07-16 21:06:21 +00:00
}
1998-11-09 16:40:38 +00:00
# ifdef EISCONN
2009-01-03 19:50:05 +01:00
if ( sys_errno = = EISCONN ) {
2009-02-26 12:34:39 +01:00
tevent_req_done ( req ) ;
2009-01-03 19:50:05 +01:00
return ;
2003-07-16 21:06:21 +00:00
}
1998-11-09 16:40:38 +00:00
# endif
2009-01-03 19:50:05 +01:00
/* real error */
2009-02-26 12:34:39 +01:00
tevent_req_nterror ( req , map_nt_error_from_unix ( sys_errno ) ) ;
2009-01-03 19:50:05 +01:00
}
2009-02-26 12:34:39 +01:00
NTSTATUS open_socket_out_recv ( struct tevent_req * req , int * pfd )
2009-01-03 19:50:05 +01:00
{
2009-02-28 15:44:30 -05:00
struct open_socket_out_state * state =
tevent_req_data ( req , struct open_socket_out_state ) ;
2009-02-26 12:34:39 +01:00
NTSTATUS status ;
2009-01-03 19:50:05 +01:00
2009-02-26 12:34:39 +01:00
if ( tevent_req_is_nterror ( req , & status ) ) {
return status ;
2009-01-03 19:50:05 +01:00
}
* pfd = state - > fd ;
state - > fd = - 1 ;
return NT_STATUS_OK ;
}
2011-04-13 17:35:44 +02:00
/**
* @ brief open a socket
*
* @ param pss a struct sockaddr_storage defining the address to connect to
* @ param port to connect to
* @ param timeout in MILLISECONDS
* @ param pfd file descriptor returned
*
* @ return NTSTATUS code
*/
2009-01-03 19:50:05 +01:00
NTSTATUS open_socket_out ( const struct sockaddr_storage * pss , uint16_t port ,
int timeout , int * pfd )
{
TALLOC_CTX * frame = talloc_stackframe ( ) ;
2013-02-18 10:00:26 +01:00
struct tevent_context * ev ;
2009-02-26 12:34:39 +01:00
struct tevent_req * req ;
2009-01-03 19:50:05 +01:00
NTSTATUS status = NT_STATUS_NO_MEMORY ;
2013-02-18 09:10:34 +01:00
ev = samba_tevent_context_init ( frame ) ;
2009-01-03 19:50:05 +01:00
if ( ev = = NULL ) {
goto fail ;
2003-07-16 21:06:21 +00:00
}
1998-11-09 16:40:38 +00:00
2009-01-03 19:50:05 +01:00
req = open_socket_out_send ( frame , ev , pss , port , timeout ) ;
if ( req = = NULL ) {
goto fail ;
}
2009-02-26 12:34:39 +01:00
if ( ! tevent_req_poll ( req , ev ) ) {
status = NT_STATUS_INTERNAL_ERROR ;
goto fail ;
2009-01-03 19:50:05 +01:00
}
status = open_socket_out_recv ( req , pfd ) ;
fail :
TALLOC_FREE ( frame ) ;
return status ;
1998-11-09 16:40:38 +00:00
}
2009-01-04 01:45:06 +01:00
struct open_socket_out_defer_state {
2013-02-18 10:00:26 +01:00
struct tevent_context * ev ;
2009-01-04 01:45:06 +01:00
struct sockaddr_storage ss ;
uint16_t port ;
int timeout ;
int fd ;
} ;
2009-03-16 18:43:57 +01:00
static void open_socket_out_defer_waited ( struct tevent_req * subreq ) ;
2009-02-26 12:34:39 +01:00
static void open_socket_out_defer_connected ( struct tevent_req * subreq ) ;
2009-01-04 01:45:06 +01:00
2009-03-16 19:15:26 +01:00
struct tevent_req * open_socket_out_defer_send ( TALLOC_CTX * mem_ctx ,
2013-02-18 10:00:26 +01:00
struct tevent_context * ev ,
2009-03-16 19:15:26 +01:00
struct timeval wait_time ,
const struct sockaddr_storage * pss ,
uint16_t port ,
int timeout )
2009-01-04 01:45:06 +01:00
{
2009-03-16 19:15:26 +01:00
struct tevent_req * req , * subreq ;
2009-01-04 01:45:06 +01:00
struct open_socket_out_defer_state * state ;
2009-03-16 19:15:26 +01:00
req = tevent_req_create ( mem_ctx , & state ,
struct open_socket_out_defer_state ) ;
if ( req = = NULL ) {
2009-01-04 01:45:06 +01:00
return NULL ;
}
state - > ev = ev ;
state - > ss = * pss ;
state - > port = port ;
state - > timeout = timeout ;
2009-03-16 18:43:57 +01:00
subreq = tevent_wakeup_send (
state , ev ,
timeval_current_ofs ( wait_time . tv_sec , wait_time . tv_usec ) ) ;
2009-01-04 01:45:06 +01:00
if ( subreq = = NULL ) {
goto fail ;
}
2009-03-16 19:15:26 +01:00
tevent_req_set_callback ( subreq , open_socket_out_defer_waited , req ) ;
return req ;
2009-01-04 01:45:06 +01:00
fail :
2009-03-16 19:15:26 +01:00
TALLOC_FREE ( req ) ;
2009-01-04 01:45:06 +01:00
return NULL ;
}
2009-03-16 18:43:57 +01:00
static void open_socket_out_defer_waited ( struct tevent_req * subreq )
2009-01-04 01:45:06 +01:00
{
2009-03-16 19:15:26 +01:00
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct open_socket_out_defer_state * state = tevent_req_data (
req , struct open_socket_out_defer_state ) ;
2009-02-01 16:32:02 +01:00
bool ret ;
2009-01-04 01:45:06 +01:00
2009-03-16 18:43:57 +01:00
ret = tevent_wakeup_recv ( subreq ) ;
2009-01-04 01:45:06 +01:00
TALLOC_FREE ( subreq ) ;
2009-02-01 16:32:02 +01:00
if ( ! ret ) {
2009-03-16 19:15:26 +01:00
tevent_req_nterror ( req , NT_STATUS_INTERNAL_ERROR ) ;
2009-01-04 01:45:06 +01:00
return ;
}
2009-03-16 18:43:57 +01:00
subreq = open_socket_out_send ( state , state - > ev , & state - > ss ,
state - > port , state - > timeout ) ;
2009-03-16 19:15:26 +01:00
if ( tevent_req_nomem ( subreq , req ) ) {
2009-01-04 01:45:06 +01:00
return ;
}
2009-03-16 18:43:57 +01:00
tevent_req_set_callback ( subreq , open_socket_out_defer_connected , req ) ;
2009-01-04 01:45:06 +01:00
}
2009-02-26 12:34:39 +01:00
static void open_socket_out_defer_connected ( struct tevent_req * subreq )
2009-01-04 01:45:06 +01:00
{
2009-03-16 19:15:26 +01:00
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct open_socket_out_defer_state * state = tevent_req_data (
req , struct open_socket_out_defer_state ) ;
2009-01-04 01:45:06 +01:00
NTSTATUS status ;
status = open_socket_out_recv ( subreq , & state - > fd ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-03-16 19:15:26 +01:00
tevent_req_nterror ( req , status ) ;
2009-01-04 01:45:06 +01:00
return ;
}
2009-03-16 19:15:26 +01:00
tevent_req_done ( req ) ;
2009-01-04 01:45:06 +01:00
}
2009-03-16 19:15:26 +01:00
NTSTATUS open_socket_out_defer_recv ( struct tevent_req * req , int * pfd )
2009-01-04 01:45:06 +01:00
{
2009-03-16 19:15:26 +01:00
struct open_socket_out_defer_state * state = tevent_req_data (
req , struct open_socket_out_defer_state ) ;
2009-01-04 01:45:06 +01:00
NTSTATUS status ;
2009-03-16 19:15:26 +01:00
if ( tevent_req_is_nterror ( req , & status ) ) {
2009-01-04 01:45:06 +01:00
return status ;
}
* pfd = state - > fd ;
state - > fd = - 1 ;
return NT_STATUS_OK ;
}
2003-07-16 21:06:21 +00:00
/****************************************************************************
Open a connected UDP socket to host on port
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-11-24 14:16:41 +00:00
int open_udp_socket ( const char * host , int port )
{
2009-06-08 13:26:39 -07:00
struct sockaddr_storage ss ;
2001-11-24 14:16:41 +00:00
int res ;
2012-03-24 14:51:23 +01:00
socklen_t salen ;
2001-11-24 14:16:41 +00:00
2009-06-08 13:26:39 -07:00
if ( ! interpret_string_addr ( & ss , host , 0 ) ) {
DEBUG ( 10 , ( " open_udp_socket: can't resolve name %s \n " ,
host ) ) ;
return - 1 ;
}
2001-11-24 14:16:41 +00:00
2009-06-08 13:26:39 -07:00
res = socket ( ss . ss_family , SOCK_DGRAM , 0 ) ;
2001-11-24 14:16:41 +00:00
if ( res = = - 1 ) {
return - 1 ;
}
2009-06-08 13:26:39 -07:00
# if defined(HAVE_IPV6)
if ( ss . ss_family = = AF_INET6 ) {
struct sockaddr_in6 * psa6 ;
psa6 = ( struct sockaddr_in6 * ) & ss ;
psa6 - > sin6_port = htons ( port ) ;
if ( psa6 - > sin6_scope_id = = 0
& & IN6_IS_ADDR_LINKLOCAL ( & psa6 - > sin6_addr ) ) {
setup_linklocal_scope_id (
( struct sockaddr * ) & ss ) ;
}
2012-03-24 14:51:23 +01:00
salen = sizeof ( struct sockaddr_in6 ) ;
} else
2009-06-08 13:26:39 -07:00
# endif
2012-03-24 14:51:23 +01:00
if ( ss . ss_family = = AF_INET ) {
struct sockaddr_in * psa ;
psa = ( struct sockaddr_in * ) & ss ;
psa - > sin_port = htons ( port ) ;
salen = sizeof ( struct sockaddr_in ) ;
} else {
DEBUG ( 1 , ( " unknown socket family %d " , ss . ss_family ) ) ;
2012-11-25 10:19:23 +01:00
close ( res ) ;
2012-03-24 14:51:23 +01:00
return - 1 ;
}
2001-11-24 14:16:41 +00:00
2012-03-24 14:51:23 +01:00
if ( connect ( res , ( struct sockaddr * ) & ss , salen ) ) {
2001-11-24 14:16:41 +00:00
close ( res ) ;
return - 1 ;
}
return res ;
}
2007-10-12 13:38:04 -07:00
/*******************************************************************
Return the IP addr of the remote end of a socket as a string .
Optionally return the struct sockaddr_storage .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static const char * get_peer_addr_internal ( int fd ,
2007-11-03 18:41:26 -07:00
char * addr_buf ,
size_t addr_buf_len ,
2008-10-23 19:53:15 +02:00
struct sockaddr * pss ,
2007-10-12 13:38:04 -07:00
socklen_t * plength )
{
struct sockaddr_storage ss ;
socklen_t length = sizeof ( ss ) ;
2007-11-07 12:48:58 -08:00
strlcpy ( addr_buf , " 0.0.0.0 " , addr_buf_len ) ;
2007-10-12 13:38:04 -07:00
if ( fd = = - 1 ) {
return addr_buf ;
}
if ( pss = = NULL ) {
2008-10-23 19:53:15 +02:00
pss = ( struct sockaddr * ) & ss ;
2007-10-12 13:38:04 -07:00
plength = & length ;
}
if ( getpeername ( fd , ( struct sockaddr * ) pss , plength ) < 0 ) {
2010-03-26 15:14:35 +01:00
int level = ( errno = = ENOTCONN ) ? 2 : 0 ;
DEBUG ( level , ( " getpeername failed. Error was %s \n " ,
strerror ( errno ) ) ) ;
2007-10-12 13:38:04 -07:00
return addr_buf ;
}
2007-10-24 14:16:54 -07:00
print_sockaddr_len ( addr_buf ,
2007-11-03 20:27:59 -07:00
addr_buf_len ,
2007-10-12 13:38:04 -07:00
pss ,
* plength ) ;
return addr_buf ;
}
1998-11-09 16:40:38 +00:00
/*******************************************************************
2003-07-16 21:06:21 +00:00
Matchname - determine if host name matches IP address . Used to
confirm a hostname lookup to prevent spoof attacks .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-12 13:38:04 -07:00
static bool matchname ( const char * remotehost ,
2008-10-23 19:53:15 +02:00
const struct sockaddr * pss ,
2007-10-12 13:38:04 -07:00
socklen_t len )
1998-11-09 16:40:38 +00:00
{
2007-10-12 13:38:04 -07:00
struct addrinfo * res = NULL ;
struct addrinfo * ailist = NULL ;
char addr_buf [ INET6_ADDRSTRLEN ] ;
2007-10-15 16:11:48 -07:00
bool ret = interpret_string_addr_internal ( & ailist ,
remotehost ,
AI_ADDRCONFIG | AI_CANONNAME ) ;
2007-10-12 13:38:04 -07:00
2007-10-15 16:11:48 -07:00
if ( ! ret | | ailist = = NULL ) {
2007-10-12 13:38:04 -07:00
DEBUG ( 3 , ( " matchname: getaddrinfo failed for "
" name %s [%s] \n " ,
remotehost ,
gai_strerror ( ret ) ) ) ;
2007-10-10 18:25:16 -07:00
return false ;
2007-10-02 19:50:53 +00:00
}
2000-04-11 13:55:53 +00:00
/*
2007-10-12 13:38:04 -07:00
* Make sure that getaddrinfo ( ) returns the " correct " host name .
2000-04-11 13:55:53 +00:00
*/
2007-10-02 19:50:53 +00:00
2007-10-15 16:11:48 -07:00
if ( ailist - > ai_canonname = = NULL | |
( ! strequal ( remotehost , ailist - > ai_canonname ) & &
2007-10-12 13:38:04 -07:00
! strequal ( remotehost , " localhost " ) ) ) {
DEBUG ( 0 , ( " matchname: host name/name mismatch: %s != %s \n " ,
remotehost ,
2007-10-15 16:11:48 -07:00
ailist - > ai_canonname ?
ailist - > ai_canonname : " (NULL) " ) ) ;
freeaddrinfo ( ailist ) ;
2007-10-10 18:25:16 -07:00
return false ;
1998-11-09 16:40:38 +00:00
}
2007-10-02 19:50:53 +00:00
2000-04-11 13:55:53 +00:00
/* Look up the host address in the address list we just got. */
2007-10-15 16:11:48 -07:00
for ( res = ailist ; res ; res = res - > ai_next ) {
if ( ! res - > ai_addr ) {
2007-10-12 13:38:04 -07:00
continue ;
}
2008-12-02 23:29:57 -08:00
if ( sockaddr_equal ( ( const struct sockaddr * ) res - > ai_addr ,
2011-05-05 13:42:05 -07:00
( const struct sockaddr * ) pss ) ) {
2007-10-15 16:11:48 -07:00
freeaddrinfo ( ailist ) ;
2007-10-10 18:25:16 -07:00
return true ;
2007-10-12 13:38:04 -07:00
}
1998-11-09 16:40:38 +00:00
}
2007-10-02 19:50:53 +00:00
2000-04-11 13:55:53 +00:00
/*
* The host name does not map to the original host address . Perhaps
* someone has compromised a name server . More likely someone botched
* it , but that could be dangerous , too .
*/
2007-10-02 19:50:53 +00:00
2007-10-12 13:38:04 -07:00
DEBUG ( 0 , ( " matchname: host name/address mismatch: %s != %s \n " ,
2007-10-24 14:16:54 -07:00
print_sockaddr_len ( addr_buf ,
2007-10-12 13:38:04 -07:00
sizeof ( addr_buf ) ,
pss ,
len ) ,
2007-10-15 16:11:48 -07:00
ailist - > ai_canonname ? ailist - > ai_canonname : " (NULL) " ) ) ;
2007-10-12 13:38:04 -07:00
2007-10-15 16:11:48 -07:00
if ( ailist ) {
freeaddrinfo ( ailist ) ;
}
2007-10-10 18:25:16 -07:00
return false ;
2000-04-11 13:55:53 +00:00
}
2008-01-02 17:37:39 -08:00
/*******************************************************************
Deal with the singleton cache .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
struct name_addr_pair {
struct sockaddr_storage ss ;
const char * name ;
} ;
/*******************************************************************
Lookup a name / addr pair . Returns memory allocated from memcache .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static bool lookup_nc ( struct name_addr_pair * nc )
{
DATA_BLOB tmp ;
ZERO_STRUCTP ( nc ) ;
if ( ! memcache_lookup (
NULL , SINGLETON_CACHE ,
2008-10-13 05:20:26 +02:00
data_blob_string_const_null ( " get_peer_name " ) ,
2008-01-02 17:37:39 -08:00
& tmp ) ) {
return false ;
}
memcpy ( & nc - > ss , tmp . data , sizeof ( nc - > ss ) ) ;
nc - > name = ( const char * ) tmp . data + sizeof ( nc - > ss ) ;
return true ;
}
/*******************************************************************
Save a name / addr pair .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void store_nc ( const struct name_addr_pair * nc )
{
DATA_BLOB tmp ;
size_t namelen = strlen ( nc - > name ) ;
2008-01-03 10:24:45 +01:00
tmp = data_blob ( NULL , sizeof ( nc - > ss ) + namelen + 1 ) ;
2008-01-02 17:37:39 -08:00
if ( ! tmp . data ) {
return ;
}
memcpy ( tmp . data , & nc - > ss , sizeof ( nc - > ss ) ) ;
memcpy ( tmp . data + sizeof ( nc - > ss ) , nc - > name , namelen + 1 ) ;
memcache_add ( NULL , SINGLETON_CACHE ,
2008-10-13 05:20:26 +02:00
data_blob_string_const_null ( " get_peer_name " ) ,
2008-01-02 17:37:39 -08:00
tmp ) ;
2008-01-03 10:24:45 +01:00
data_blob_free ( & tmp ) ;
2008-01-02 17:37:39 -08:00
}
2007-11-07 12:48:58 -08:00
1998-11-09 16:40:38 +00:00
/*******************************************************************
2003-07-16 21:06:21 +00:00
Return the IP addr of the remote end of a socket as a string .
1998-11-09 16:40:38 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2003-07-16 21:06:21 +00:00
2007-11-03 18:41:26 -07:00
const char * get_peer_addr ( int fd , char * addr , size_t addr_len )
1998-11-09 16:40:38 +00:00
{
2007-11-03 18:41:26 -07:00
return get_peer_addr_internal ( fd , addr , addr_len , NULL , NULL ) ;
1998-11-09 16:40:38 +00:00
}
2000-01-03 19:19:48 +00:00
2011-06-08 14:50:20 +02:00
int get_remote_hostname ( const struct tsocket_address * remote_address ,
char * * name ,
TALLOC_CTX * mem_ctx )
{
char name_buf [ MAX_DNS_NAME_LENGTH ] ;
char tmp_name [ MAX_DNS_NAME_LENGTH ] ;
struct name_addr_pair nc ;
struct sockaddr_storage ss ;
2011-07-06 09:37:04 +02:00
ssize_t len ;
2011-06-08 14:50:20 +02:00
int rc ;
if ( ! lp_hostname_lookups ( ) ) {
nc . name = tsocket_address_inet_addr_string ( remote_address ,
mem_ctx ) ;
if ( nc . name = = NULL ) {
return - 1 ;
}
len = tsocket_address_bsd_sockaddr ( remote_address ,
( struct sockaddr * ) & nc . ss ,
sizeof ( struct sockaddr_storage ) ) ;
if ( len < 0 ) {
return - 1 ;
}
store_nc ( & nc ) ;
lookup_nc ( & nc ) ;
if ( nc . name = = NULL ) {
* name = talloc_strdup ( mem_ctx , " UNKNOWN " ) ;
} else {
* name = talloc_strdup ( mem_ctx , nc . name ) ;
}
return 0 ;
}
lookup_nc ( & nc ) ;
ZERO_STRUCT ( ss ) ;
len = tsocket_address_bsd_sockaddr ( remote_address ,
( struct sockaddr * ) & ss ,
sizeof ( struct sockaddr_storage ) ) ;
if ( len < 0 ) {
return - 1 ;
}
/* it might be the same as the last one - save some DNS work */
if ( sockaddr_equal ( ( struct sockaddr * ) & ss , ( struct sockaddr * ) & nc . ss ) ) {
if ( nc . name = = NULL ) {
* name = talloc_strdup ( mem_ctx , " UNKNOWN " ) ;
} else {
* name = talloc_strdup ( mem_ctx , nc . name ) ;
}
return 0 ;
}
/* Look up the remote host name. */
rc = sys_getnameinfo ( ( struct sockaddr * ) & ss ,
len ,
name_buf ,
sizeof ( name_buf ) ,
NULL ,
0 ,
0 ) ;
if ( rc < 0 ) {
char * p ;
p = tsocket_address_inet_addr_string ( remote_address , mem_ctx ) ;
if ( p = = NULL ) {
return - 1 ;
}
DEBUG ( 1 , ( " getnameinfo failed for %s with error %s \n " ,
p ,
gai_strerror ( rc ) ) ) ;
strlcpy ( name_buf , p , sizeof ( name_buf ) ) ;
2011-07-06 09:39:08 +02:00
TALLOC_FREE ( p ) ;
2011-06-08 14:50:20 +02:00
} else {
if ( ! matchname ( name_buf , ( struct sockaddr * ) & ss , len ) ) {
DEBUG ( 0 , ( " matchname failed on %s \n " , name_buf ) ) ;
strlcpy ( name_buf , " UNKNOWN " , sizeof ( name_buf ) ) ;
}
}
strlcpy ( tmp_name , name_buf , sizeof ( tmp_name ) ) ;
alpha_strcpy ( name_buf , tmp_name , " _-. " , sizeof ( name_buf ) ) ;
if ( strstr ( name_buf , " .. " ) ) {
strlcpy ( name_buf , " UNKNOWN " , sizeof ( name_buf ) ) ;
}
nc . name = name_buf ;
nc . ss = ss ;
store_nc ( & nc ) ;
lookup_nc ( & nc ) ;
if ( nc . name = = NULL ) {
* name = talloc_strdup ( mem_ctx , " UNKOWN " ) ;
} else {
* name = talloc_strdup ( mem_ctx , nc . name ) ;
}
return 0 ;
}
2002-01-13 11:13:54 +00:00
/*******************************************************************
Create protected unix domain socket .
2003-02-07 21:59:51 +00:00
Some unixes cannot set permissions on a ux - dom - sock , so we
2002-01-13 11:13:54 +00:00
have to make sure that the directory contains the protection
2003-02-07 21:59:51 +00:00
permissions instead .
2002-01-13 11:13:54 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2003-02-07 21:59:51 +00:00
2002-01-13 11:13:54 +00:00
int create_pipe_sock ( const char * socket_dir ,
2002-07-15 10:35:28 +00:00
const char * socket_name ,
mode_t dir_perms )
2002-01-13 11:13:54 +00:00
{
2002-09-25 15:19:00 +00:00
# ifdef HAVE_UNIXSOCKET
2003-02-07 21:59:51 +00:00
struct sockaddr_un sunaddr ;
2013-01-09 09:02:54 +01:00
bool ok ;
2013-06-17 17:02:56 -07:00
int sock = - 1 ;
mode_t old_umask ;
2007-11-11 21:45:55 -08:00
char * path = NULL ;
2007-10-02 19:50:53 +00:00
2013-06-17 17:02:56 -07:00
old_umask = umask ( 0 ) ;
2013-01-09 09:02:54 +01:00
ok = directory_create_or_exist_strict ( socket_dir ,
sec_initial_uid ( ) ,
dir_perms ) ;
if ( ! ok ) {
2013-06-17 17:02:56 -07:00
goto out_close ;
2003-02-07 21:59:51 +00:00
}
2007-10-02 19:50:53 +00:00
2003-02-07 21:59:51 +00:00
/* Create the socket file */
sock = socket ( AF_UNIX , SOCK_STREAM , 0 ) ;
2007-10-02 19:50:53 +00:00
2003-02-07 21:59:51 +00:00
if ( sock = = - 1 ) {
2007-11-11 21:45:55 -08:00
DEBUG ( 0 , ( " create_pipe_sock: socket error %s \n " ,
strerror ( errno ) ) ) ;
goto out_close ;
2003-02-07 21:59:51 +00:00
}
2007-10-02 19:50:53 +00:00
2008-02-25 15:24:49 +01:00
if ( asprintf ( & path , " %s/%s " , socket_dir , socket_name ) = = - 1 ) {
2007-11-11 21:45:55 -08:00
goto out_close ;
}
2007-10-02 19:50:53 +00:00
2003-02-07 21:59:51 +00:00
unlink ( path ) ;
memset ( & sunaddr , 0 , sizeof ( sunaddr ) ) ;
sunaddr . sun_family = AF_UNIX ;
2007-11-07 12:48:58 -08:00
strlcpy ( sunaddr . sun_path , path , sizeof ( sunaddr . sun_path ) ) ;
2007-10-02 19:50:53 +00:00
2003-02-07 21:59:51 +00:00
if ( bind ( sock , ( struct sockaddr * ) & sunaddr , sizeof ( sunaddr ) ) = = - 1 ) {
DEBUG ( 0 , ( " bind failed on pipe socket %s: %s \n " , path ,
strerror ( errno ) ) ) ;
goto out_close ;
}
2007-10-02 19:50:53 +00:00
2007-11-11 21:45:55 -08:00
SAFE_FREE ( path ) ;
2013-06-17 17:02:56 -07:00
umask ( old_umask ) ;
2003-02-07 21:59:51 +00:00
return sock ;
out_close :
2007-11-11 21:45:55 -08:00
SAFE_FREE ( path ) ;
2008-02-18 10:43:46 +11:00
if ( sock ! = - 1 )
close ( sock ) ;
2003-02-07 21:59:51 +00:00
2013-06-17 17:02:56 -07:00
umask ( old_umask ) ;
2003-02-07 21:59:51 +00:00
return - 1 ;
2002-09-25 15:19:00 +00:00
# else
DEBUG ( 0 , ( " create_pipe_sock: No Unix sockets on this system \n " ) ) ;
return - 1 ;
# endif /* HAVE_UNIXSOCKET */
2002-01-13 11:13:54 +00:00
}
2007-10-10 18:25:16 -07:00
2007-10-15 16:11:48 -07:00
/****************************************************************************
Get my own canonical name , including domain .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-11-08 17:25:45 -08:00
const char * get_mydnsfullname ( void )
2007-10-15 16:11:48 -07:00
{
2008-01-02 17:37:39 -08:00
struct addrinfo * res = NULL ;
char my_hostname [ HOST_NAME_MAX ] ;
bool ret ;
DATA_BLOB tmp ;
2007-10-15 16:11:48 -07:00
2008-01-02 17:37:39 -08:00
if ( memcache_lookup ( NULL , SINGLETON_CACHE ,
2008-10-13 05:20:26 +02:00
data_blob_string_const_null ( " get_mydnsfullname " ) ,
2008-01-02 17:37:39 -08:00
& tmp ) ) {
SMB_ASSERT ( tmp . length > 0 ) ;
return ( const char * ) tmp . data ;
}
2007-10-15 16:11:48 -07:00
2008-01-02 17:37:39 -08:00
/* get my host name */
if ( gethostname ( my_hostname , sizeof ( my_hostname ) ) = = - 1 ) {
DEBUG ( 0 , ( " get_mydnsfullname: gethostname failed \n " ) ) ;
return NULL ;
}
2007-10-15 16:11:48 -07:00
2008-01-02 17:37:39 -08:00
/* Ensure null termination. */
my_hostname [ sizeof ( my_hostname ) - 1 ] = ' \0 ' ;
2007-10-15 16:11:48 -07:00
2008-01-02 17:37:39 -08:00
ret = interpret_string_addr_internal ( & res ,
2007-11-08 17:25:45 -08:00
my_hostname ,
2008-01-02 17:37:39 -08:00
AI_ADDRCONFIG | AI_CANONNAME ) ;
2007-10-15 16:11:48 -07:00
2008-01-02 17:37:39 -08:00
if ( ! ret | | res = = NULL ) {
DEBUG ( 3 , ( " get_mydnsfullname: getaddrinfo failed for "
" name %s [%s] \n " ,
my_hostname ,
gai_strerror ( ret ) ) ) ;
return NULL ;
}
2007-10-15 16:11:48 -07:00
2008-01-02 17:37:39 -08:00
/*
* Make sure that getaddrinfo ( ) returns the " correct " host name .
*/
2007-10-15 16:11:48 -07:00
2008-01-02 17:37:39 -08:00
if ( res - > ai_canonname = = NULL ) {
DEBUG ( 3 , ( " get_mydnsfullname: failed to get "
" canonical name for %s \n " ,
my_hostname ) ) ;
2007-10-15 16:11:48 -07:00
freeaddrinfo ( res ) ;
2008-01-02 17:37:39 -08:00
return NULL ;
}
/* This copies the data, so we must do a lookup
* afterwards to find the value to return .
*/
memcache_add ( NULL , SINGLETON_CACHE ,
2008-10-13 05:20:26 +02:00
data_blob_string_const_null ( " get_mydnsfullname " ) ,
data_blob_string_const_null ( res - > ai_canonname ) ) ;
2008-01-02 17:37:39 -08:00
if ( ! memcache_lookup ( NULL , SINGLETON_CACHE ,
2008-10-13 05:20:26 +02:00
data_blob_string_const_null ( " get_mydnsfullname " ) ,
2008-01-02 17:37:39 -08:00
& tmp ) ) {
2008-01-21 18:01:55 +01:00
tmp = data_blob_talloc ( talloc_tos ( ) , res - > ai_canonname ,
strlen ( res - > ai_canonname ) + 1 ) ;
2007-10-15 16:11:48 -07:00
}
2008-01-02 17:37:39 -08:00
2008-01-21 13:18:38 +01:00
freeaddrinfo ( res ) ;
2008-01-02 17:37:39 -08:00
return ( const char * ) tmp . data ;
2007-10-15 16:11:48 -07:00
}
2011-03-24 12:11:02 -07:00
/************************************************************
Is this my ip address ?
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static bool is_my_ipaddr ( const char * ipaddr_str )
{
struct sockaddr_storage ss ;
struct iface_struct * nics ;
int i , n ;
if ( ! interpret_string_addr ( & ss , ipaddr_str , AI_NUMERICHOST ) ) {
return false ;
}
2011-05-16 16:08:48 -07:00
if ( is_zero_addr ( & ss ) ) {
return false ;
2011-03-24 12:11:02 -07:00
}
2011-05-16 16:08:48 -07:00
if ( ismyaddr ( ( struct sockaddr * ) & ss ) | |
is_loopback_addr ( ( struct sockaddr * ) & ss ) ) {
return true ;
2011-03-24 12:11:02 -07:00
}
n = get_interfaces ( talloc_tos ( ) , & nics ) ;
for ( i = 0 ; i < n ; i + + ) {
if ( sockaddr_equal ( ( struct sockaddr * ) & nics [ i ] . ip , ( struct sockaddr * ) & ss ) ) {
TALLOC_FREE ( nics ) ;
return true ;
}
}
TALLOC_FREE ( nics ) ;
return false ;
}
2007-10-10 18:25:16 -07:00
/************************************************************
2007-10-15 16:11:48 -07:00
Is this my name ?
2007-10-10 18:25:16 -07:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
bool is_myname_or_ipaddr ( const char * s )
{
2007-11-08 17:25:45 -08:00
TALLOC_CTX * ctx = talloc_tos ( ) ;
char * name = NULL ;
const char * dnsname ;
char * servername = NULL ;
2007-10-10 18:25:16 -07:00
2007-10-15 16:11:48 -07:00
if ( ! s ) {
2007-10-10 18:25:16 -07:00
return false ;
2007-10-12 21:50:41 -07:00
}
2007-10-10 18:25:16 -07:00
2007-10-15 16:11:48 -07:00
/* Santize the string from '\\name' */
2007-11-08 17:25:45 -08:00
name = talloc_strdup ( ctx , s ) ;
if ( ! name ) {
return false ;
}
2007-10-10 18:25:16 -07:00
2007-10-15 16:11:48 -07:00
servername = strrchr_m ( name , ' \\ ' ) ;
if ( ! servername ) {
2007-10-10 18:25:16 -07:00
servername = name ;
2007-10-15 16:11:48 -07:00
} else {
2007-10-10 18:25:16 -07:00
servername + + ;
2007-10-15 16:11:48 -07:00
}
2007-10-10 18:25:16 -07:00
2007-10-15 16:11:48 -07:00
/* Optimize for the common case */
2011-06-09 15:31:03 +10:00
if ( strequal ( servername , lp_netbios_name ( ) ) ) {
2007-10-10 18:25:16 -07:00
return true ;
2007-10-15 16:11:48 -07:00
}
2007-10-10 18:25:16 -07:00
2007-10-15 16:11:48 -07:00
/* Check for an alias */
if ( is_myname ( servername ) ) {
2007-10-10 18:25:16 -07:00
return true ;
2007-10-15 16:11:48 -07:00
}
2007-10-10 18:25:16 -07:00
2007-10-15 16:11:48 -07:00
/* Check for loopback */
if ( strequal ( servername , " 127.0.0.1 " ) | |
strequal ( servername , " ::1 " ) ) {
2007-10-10 18:25:16 -07:00
return true ;
2007-10-15 16:11:48 -07:00
}
2007-10-10 18:25:16 -07:00
2007-10-15 16:11:48 -07:00
if ( strequal ( servername , " localhost " ) ) {
2007-10-10 18:25:16 -07:00
return true ;
2007-10-15 16:11:48 -07:00
}
2007-10-10 18:25:16 -07:00
2007-10-15 16:11:48 -07:00
/* Maybe it's my dns name */
2007-11-08 17:25:45 -08:00
dnsname = get_mydnsfullname ( ) ;
if ( dnsname & & strequal ( servername , dnsname ) ) {
return true ;
2007-10-15 16:11:48 -07:00
}
2007-10-10 18:25:16 -07:00
2007-10-15 16:11:48 -07:00
/* Maybe its an IP address? */
if ( is_ipaddress ( servername ) ) {
2011-03-24 12:11:02 -07:00
return is_my_ipaddr ( servername ) ;
}
2007-10-10 18:25:16 -07:00
2011-03-24 12:11:02 -07:00
/* Handle possible CNAME records - convert to an IP addr. list. */
{
/* Use DNS to resolve the name, check all addresses. */
struct addrinfo * p = NULL ;
struct addrinfo * res = NULL ;
2009-04-08 12:47:10 +02:00
2011-03-24 12:11:02 -07:00
if ( ! interpret_string_addr_internal ( & res ,
servername ,
AI_ADDRCONFIG ) ) {
2007-10-15 16:11:48 -07:00
return false ;
}
2007-10-10 18:25:16 -07:00
2011-03-24 12:11:02 -07:00
for ( p = res ; p ; p = p - > ai_next ) {
char addr [ INET6_ADDRSTRLEN ] ;
struct sockaddr_storage ss ;
ZERO_STRUCT ( ss ) ;
memcpy ( & ss , p - > ai_addr , p - > ai_addrlen ) ;
print_sockaddr ( addr ,
sizeof ( addr ) ,
& ss ) ;
if ( is_my_ipaddr ( addr ) ) {
freeaddrinfo ( res ) ;
2007-10-10 18:25:16 -07:00
return true ;
}
}
2011-03-24 12:11:02 -07:00
freeaddrinfo ( res ) ;
2007-10-10 18:25:16 -07:00
}
2007-10-15 16:11:48 -07:00
/* No match */
2007-10-10 18:25:16 -07:00
return false ;
}
2009-04-05 19:58:20 +02:00
2009-04-26 00:01:43 +02:00
struct getaddrinfo_state {
const char * node ;
const char * service ;
const struct addrinfo * hints ;
struct addrinfo * res ;
int ret ;
} ;
static void getaddrinfo_do ( void * private_data ) ;
static void getaddrinfo_done ( struct tevent_req * subreq ) ;
struct tevent_req * getaddrinfo_send ( TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
struct fncall_context * ctx ,
const char * node ,
const char * service ,
const struct addrinfo * hints )
{
struct tevent_req * req , * subreq ;
struct getaddrinfo_state * state ;
req = tevent_req_create ( mem_ctx , & state , struct getaddrinfo_state ) ;
if ( req = = NULL ) {
return NULL ;
}
state - > node = node ;
state - > service = service ;
state - > hints = hints ;
subreq = fncall_send ( state , ev , ctx , getaddrinfo_do , state ) ;
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
}
tevent_req_set_callback ( subreq , getaddrinfo_done , req ) ;
return req ;
}
static void getaddrinfo_do ( void * private_data )
{
struct getaddrinfo_state * state =
( struct getaddrinfo_state * ) private_data ;
state - > ret = getaddrinfo ( state - > node , state - > service , state - > hints ,
& state - > res ) ;
}
static void getaddrinfo_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
int ret , err ;
ret = fncall_recv ( subreq , & err ) ;
TALLOC_FREE ( subreq ) ;
if ( ret = = - 1 ) {
tevent_req_error ( req , err ) ;
return ;
}
tevent_req_done ( req ) ;
}
int getaddrinfo_recv ( struct tevent_req * req , struct addrinfo * * res )
{
struct getaddrinfo_state * state = tevent_req_data (
req , struct getaddrinfo_state ) ;
int err ;
if ( tevent_req_is_unix_error ( req , & err ) ) {
switch ( err ) {
case ENOMEM :
return EAI_MEMORY ;
default :
return EAI_FAIL ;
}
}
if ( state - > ret = = 0 ) {
* res = state - > res ;
}
return state - > ret ;
}
2011-02-07 16:55:16 +01:00
int poll_one_fd ( int fd , int events , int timeout , int * revents )
{
2013-12-04 23:31:10 +01:00
struct pollfd pfd ;
2011-02-07 16:55:16 +01:00
int ret ;
2013-12-04 23:31:10 +01:00
pfd . fd = fd ;
pfd . events = events ;
2011-02-07 16:55:16 +01:00
2013-12-04 23:31:10 +01:00
ret = poll ( & pfd , 1 , timeout ) ;
2011-02-07 16:55:16 +01:00
/*
* Assign whatever poll did , even in the ret < = 0 case .
*/
2013-12-04 23:31:10 +01:00
* revents = pfd . revents ;
2011-02-07 16:55:16 +01:00
return ret ;
}
2011-02-08 17:33:59 +01:00
int poll_intr_one_fd ( int fd , int events , int timeout , int * revents )
{
struct pollfd pfd ;
int ret ;
pfd . fd = fd ;
pfd . events = events ;
ret = sys_poll_intr ( & pfd , 1 , timeout ) ;
if ( ret < = 0 ) {
* revents = 0 ;
return ret ;
}
* revents = pfd . revents ;
return 1 ;
}