2011-06-30 10:46:48 +02:00
/*
* Unix SMB / CIFS implementation .
*
* RPC Socket Helper
*
* Copyright ( c ) 2011 Andreas Schneider < asn @ samba . org >
*
* 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 3 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 , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
# include "ntdomain.h"
# include "../lib/tsocket/tsocket.h"
# include "librpc/rpc/dcerpc_ep.h"
# include "rpc_server/rpc_server.h"
# include "rpc_server/rpc_sock_helper.h"
2019-01-28 10:57:53 +01:00
# include "lib/server_prefork.h"
2011-06-30 10:46:48 +02:00
2019-02-05 19:27:58 +01:00
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_RPC_SRV
2019-01-22 16:32:51 +01:00
NTSTATUS dcesrv_create_ncacn_ip_tcp_sockets (
const struct ndr_interface_table * iface ,
struct dcerpc_binding_vector * bvec ,
uint16_t port ,
2019-01-28 10:57:53 +01:00
struct pf_listen_fd * listen_fd ,
2019-01-22 16:32:51 +01:00
int * listen_fd_size )
2011-06-30 10:46:48 +02:00
{
uint32_t num_ifs = iface_count ( ) ;
uint32_t i ;
uint16_t p = port ;
TALLOC_CTX * tmp_ctx ;
NTSTATUS status = NT_STATUS_UNSUCCESSFUL ;
int rc ;
tmp_ctx = talloc_stackframe ( ) ;
if ( tmp_ctx = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
if ( lp_interfaces ( ) & & lp_bind_interfaces_only ( ) ) {
/*
* We have been given an interfaces line , and been told to only
* bind to those interfaces . Create a socket per interface and
* bind to only these .
*/
/* Now open a listen socket for each of the interfaces. */
for ( i = 0 ; i < num_ifs ; i + + ) {
const struct sockaddr_storage * ifss =
iface_n_sockaddr_storage ( i ) ;
struct tsocket_address * bind_addr ;
const char * addr ;
int fd ;
2019-06-04 16:16:36 +02:00
status = dcesrv_create_ncacn_ip_tcp_socket ( ifss ,
& p ,
& fd ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2011-06-30 10:46:48 +02:00
goto done ;
}
2019-01-28 10:57:53 +01:00
listen_fd [ * listen_fd_size ] . fd = fd ;
listen_fd [ * listen_fd_size ] . fd_data = NULL ;
2011-06-30 10:46:48 +02:00
( * listen_fd_size ) + + ;
if ( bvec ! = NULL ) {
rc = tsocket_address_bsd_from_sockaddr ( tmp_ctx ,
2013-06-11 19:37:47 +02:00
( const struct sockaddr * ) ifss ,
2011-06-30 10:46:48 +02:00
sizeof ( struct sockaddr_storage ) ,
& bind_addr ) ;
if ( rc < 0 ) {
close ( fd ) ;
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
addr = tsocket_address_inet_addr_string ( bind_addr ,
tmp_ctx ) ;
if ( addr = = NULL ) {
close ( fd ) ;
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
status = dcerpc_binding_vector_add_port ( iface ,
bvec ,
addr ,
p ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
close ( fd ) ;
goto done ;
}
}
}
} else {
2012-07-27 12:31:28 +10:00
const char * sock_addr ;
2011-06-30 10:46:48 +02:00
const char * sock_ptr ;
char * sock_tok ;
2018-11-20 15:56:14 +01:00
# ifdef HAVE_IPV6
2012-07-27 12:31:28 +10:00
sock_addr = " ::,0.0.0.0 " ;
2011-06-30 10:46:48 +02:00
# else
2012-07-27 12:31:28 +10:00
sock_addr = " 0.0.0.0 " ;
2011-06-30 10:46:48 +02:00
# endif
for ( sock_ptr = sock_addr ;
next_token_talloc ( talloc_tos ( ) , & sock_ptr , & sock_tok , " \t , " ) ;
) {
struct sockaddr_storage ss ;
int fd ;
/* open an incoming socket */
if ( ! interpret_string_addr ( & ss ,
sock_tok ,
AI_NUMERICHOST | AI_PASSIVE ) ) {
continue ;
}
2019-06-04 16:16:36 +02:00
status = dcesrv_create_ncacn_ip_tcp_socket ( & ss ,
& p ,
& fd ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2011-06-30 10:46:48 +02:00
goto done ;
}
2019-01-28 10:57:53 +01:00
listen_fd [ * listen_fd_size ] . fd = fd ;
listen_fd [ * listen_fd_size ] . fd_data = NULL ;
2011-06-30 10:46:48 +02:00
( * listen_fd_size ) + + ;
if ( bvec ! = NULL ) {
status = dcerpc_binding_vector_add_port ( iface ,
bvec ,
2015-04-30 11:20:58 +02:00
sock_tok ,
2011-06-30 10:46:48 +02:00
p ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
close ( fd ) ;
2013-11-01 09:02:48 +01:00
goto done ;
2011-06-30 10:46:48 +02:00
}
}
}
}
status = NT_STATUS_OK ;
done :
talloc_free ( tmp_ctx ) ;
return status ;
}
2019-01-22 16:32:51 +01:00
NTSTATUS dcesrv_setup_ncacn_ip_tcp_sockets ( struct tevent_context * ev_ctx ,
struct messaging_context * msg_ctx ,
const struct ndr_interface_table * iface ,
struct dcerpc_binding_vector * bvec ,
uint16_t port )
2011-06-30 10:46:48 +02:00
{
uint32_t num_ifs = iface_count ( ) ;
uint32_t i ;
2019-06-04 16:45:07 +02:00
uint16_t p = port ;
2011-06-30 10:46:48 +02:00
TALLOC_CTX * tmp_ctx ;
NTSTATUS status = NT_STATUS_UNSUCCESSFUL ;
int rc ;
tmp_ctx = talloc_stackframe ( ) ;
if ( tmp_ctx = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
if ( lp_interfaces ( ) & & lp_bind_interfaces_only ( ) ) {
/*
* We have been given an interfaces line , and been told to only
* bind to those interfaces . Create a socket per interface and
* bind to only these .
*/
/* Now open a listen socket for each of the interfaces. */
for ( i = 0 ; i < num_ifs ; i + + ) {
const struct sockaddr_storage * ifss =
iface_n_sockaddr_storage ( i ) ;
struct tsocket_address * bind_addr ;
const char * addr ;
2019-06-04 16:45:07 +02:00
status = dcesrv_setup_ncacn_ip_tcp_socket ( ev_ctx ,
msg_ctx ,
ifss ,
& p ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2011-06-30 10:46:48 +02:00
goto done ;
}
if ( bvec ! = NULL ) {
rc = tsocket_address_bsd_from_sockaddr ( tmp_ctx ,
2013-06-11 19:37:47 +02:00
( const struct sockaddr * ) ifss ,
2011-06-30 10:46:48 +02:00
sizeof ( struct sockaddr_storage ) ,
& bind_addr ) ;
if ( rc < 0 ) {
2013-11-01 09:02:48 +01:00
status = NT_STATUS_NO_MEMORY ;
goto done ;
2011-06-30 10:46:48 +02:00
}
addr = tsocket_address_inet_addr_string ( bind_addr ,
tmp_ctx ) ;
if ( addr = = NULL ) {
2013-11-01 09:02:48 +01:00
status = NT_STATUS_NO_MEMORY ;
goto done ;
2011-06-30 10:46:48 +02:00
}
status = dcerpc_binding_vector_add_port ( iface ,
bvec ,
addr ,
p ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2013-11-01 09:02:48 +01:00
goto done ;
2011-06-30 10:46:48 +02:00
}
}
}
} else {
2012-07-27 12:31:28 +10:00
const char * sock_addr ;
2011-06-30 10:46:48 +02:00
const char * sock_ptr ;
char * sock_tok ;
2018-11-20 15:56:14 +01:00
# ifdef HAVE_IPV6
2012-07-27 12:31:28 +10:00
sock_addr = " ::,0.0.0.0 " ;
2011-06-30 10:46:48 +02:00
# else
2012-07-27 12:31:28 +10:00
sock_addr = " 0.0.0.0 " ;
2011-06-30 10:46:48 +02:00
# endif
for ( sock_ptr = sock_addr ;
next_token_talloc ( talloc_tos ( ) , & sock_ptr , & sock_tok , " \t , " ) ;
) {
struct sockaddr_storage ss ;
/* open an incoming socket */
if ( ! interpret_string_addr ( & ss ,
sock_tok ,
AI_NUMERICHOST | AI_PASSIVE ) ) {
continue ;
}
2019-06-04 16:45:07 +02:00
status = dcesrv_setup_ncacn_ip_tcp_socket ( ev_ctx ,
msg_ctx ,
& ss ,
& p ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2013-11-01 09:02:48 +01:00
goto done ;
2011-06-30 10:46:48 +02:00
}
if ( bvec ! = NULL ) {
status = dcerpc_binding_vector_add_port ( iface ,
bvec ,
sock_tok ,
p ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2013-11-01 09:02:48 +01:00
goto done ;
2011-06-30 10:46:48 +02:00
}
}
}
}
status = NT_STATUS_OK ;
done :
talloc_free ( tmp_ctx ) ;
return status ;
}
/* vim: set ts=8 sw=8 noet cindent syntax=c.doxygen: */