2010-06-18 11:00:38 -04:00
/*
Unix SMB / Netbios implementation .
Generic infrstructure for RPC Daemons
Copyright ( C ) Simo Sorce 2010
2011-07-18 22:26:31 +10:00
Copyright ( C ) Andrew Bartlett 2011
2011-07-21 13:13:35 +02:00
Copyright ( C ) Andreas Schneider 2011
2010-06-18 11:00:38 -04: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
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"
2011-07-21 11:02:59 -04:00
# include "rpc_server/rpc_pipes.h"
2010-06-18 11:00:38 -04:00
# include "rpc_server/rpc_server.h"
2011-08-10 16:33:22 -04:00
# include "rpc_server/rpc_config.h"
2010-06-18 10:40:19 -04:00
# include "rpc_dce.h"
# include "librpc/gen_ndr/netlogon.h"
2011-02-09 14:22:16 +11:00
# include "librpc/gen_ndr/auth.h"
2010-06-18 10:55:44 -04:00
# include "lib/tsocket/tsocket.h"
2010-06-18 10:56:58 -04:00
# include "libcli/named_pipe_auth/npa_tstream.h"
2011-02-09 14:22:16 +11:00
# include "../auth/auth_sam_reply.h"
2011-03-24 12:08:15 +01:00
# include "auth.h"
2011-04-29 16:40:24 +02:00
# include "rpc_server/rpc_ncacn_np.h"
2011-04-29 23:32:28 +02:00
# include "rpc_server/srv_pipe_hnd.h"
2011-05-02 13:27:45 +02:00
# include "rpc_server/srv_pipe.h"
2010-06-18 10:40:19 -04:00
/* Creates a pipes_struct and initializes it with the information
* sent from the client */
2013-09-25 16:33:00 +02:00
int make_server_pipes_struct ( TALLOC_CTX * mem_ctx ,
struct messaging_context * msg_ctx ,
const char * pipe_name ,
enum dcerpc_transport_t transport ,
const struct tsocket_address * remote_address ,
2017-03-10 12:38:33 +13:00
const struct tsocket_address * local_address ,
2013-09-25 16:33:00 +02:00
struct auth_session_info * session_info ,
struct pipes_struct * * _p ,
int * perrno )
2010-06-18 10:40:19 -04:00
{
struct pipes_struct * p ;
2011-07-21 11:02:59 -04:00
int ret ;
2011-07-21 19:54:01 +02:00
ret = make_base_pipes_struct ( mem_ctx , msg_ctx , pipe_name ,
2011-07-21 11:02:59 -04:00
transport , RPC_LITTLE_ENDIAN ,
remote_address , local_address , & p ) ;
if ( ret ) {
* perrno = ret ;
2010-06-18 10:40:19 -04:00
return - 1 ;
}
2011-07-18 12:23:04 +10:00
if ( session_info - > unix_token & & session_info - > unix_info & & session_info - > security_token ) {
/* Don't call create_local_token(), we already have the full details here */
2011-07-18 13:10:30 +10:00
p - > session_info = talloc_steal ( p , session_info ) ;
2011-07-18 12:23:04 +10:00
} else {
2012-07-15 14:31:01 +10:00
DEBUG ( 0 , ( " Supplied session_info in make_server_pipes_struct was incomplete! " ) ) ;
* perrno = EINVAL ;
return - 1 ;
2010-06-18 10:40:19 -04:00
}
* _p = p ;
return 0 ;
}
2010-06-18 10:55:44 -04:00
/* Start listening on the appropriate unix socket and setup all is needed to
* dispatch requests to the pipes rpc implementation */
2010-06-18 11:00:38 -04:00
2011-02-15 12:57:26 +01:00
struct dcerpc_ncacn_listen_state {
2011-02-14 10:29:14 +01:00
struct ndr_syntax_id syntax_id ;
2010-06-18 11:00:38 -04:00
int fd ;
2011-02-15 12:57:26 +01:00
union {
char * name ;
uint16_t port ;
} ep ;
2011-02-14 10:29:14 +01:00
struct tevent_context * ev_ctx ;
struct messaging_context * msg_ctx ;
2011-03-14 12:29:49 +01:00
dcerpc_ncacn_disconnect_fn disconnect_fn ;
2010-06-18 11:00:38 -04:00
} ;
static void named_pipe_listener ( struct tevent_context * ev ,
struct tevent_fd * fde ,
uint16_t flags ,
2010-05-20 09:16:29 -04:00
void * private_data ) ;
2010-06-18 11:00:38 -04:00
2011-04-21 15:05:12 -04:00
int create_named_pipe_socket ( const char * pipe_name )
2010-06-18 11:00:38 -04:00
{
2011-04-21 15:05:12 -04:00
char * np_dir = NULL ;
int fd = - 1 ;
2011-07-18 11:42:43 +02:00
2011-05-17 10:32:38 +02:00
/*
* As lp_ncalrpc_dir ( ) should have 0755 , but
* lp_ncalrpc_dir ( ) / np should have 0700 , we need to
* create lp_ncalrpc_dir ( ) first .
*/
2014-07-27 19:18:09 +02:00
if ( ! directory_create_or_exist ( lp_ncalrpc_dir ( ) , 0755 ) ) {
2011-05-17 10:32:38 +02:00
DEBUG ( 0 , ( " Failed to create pipe directory %s - %s \n " ,
lp_ncalrpc_dir ( ) , strerror ( errno ) ) ) ;
goto out ;
}
2011-04-21 15:05:12 -04:00
np_dir = talloc_asprintf ( talloc_tos ( ) , " %s/np " , lp_ncalrpc_dir ( ) ) ;
2010-06-18 11:00:38 -04:00
if ( ! np_dir ) {
DEBUG ( 0 , ( " Out of memory \n " ) ) ;
goto out ;
}
2013-01-08 14:21:23 +01:00
if ( ! directory_create_or_exist_strict ( np_dir , geteuid ( ) , 0700 ) ) {
2011-05-17 10:31:14 +02:00
DEBUG ( 0 , ( " Failed to create pipe directory %s - %s \n " ,
np_dir , strerror ( errno ) ) ) ;
goto out ;
}
2011-04-21 15:05:12 -04:00
fd = create_pipe_sock ( np_dir , pipe_name , 0700 ) ;
if ( fd = = - 1 ) {
2010-06-18 11:00:38 -04:00
DEBUG ( 0 , ( " Failed to create pipe socket! [%s/%s] \n " ,
np_dir , pipe_name ) ) ;
goto out ;
}
2011-04-21 15:05:12 -04:00
2017-02-18 09:02:23 +13:00
DEBUG ( 10 , ( " Opened pipe socket fd %d for %s \n " , fd , pipe_name ) ) ;
2011-04-21 15:05:12 -04:00
out :
2011-07-21 11:11:51 +02:00
talloc_free ( np_dir ) ;
2011-04-21 15:05:12 -04:00
return fd ;
}
bool setup_named_pipe_socket ( const char * pipe_name ,
struct tevent_context * ev_ctx ,
struct messaging_context * msg_ctx )
{
struct dcerpc_ncacn_listen_state * state ;
struct tevent_fd * fde ;
2011-08-29 09:49:22 +02:00
int rc ;
2011-04-21 15:05:12 -04:00
state = talloc ( ev_ctx , struct dcerpc_ncacn_listen_state ) ;
if ( ! state ) {
DEBUG ( 0 , ( " Out of memory \n " ) ) ;
return false ;
}
state - > ep . name = talloc_strdup ( state , pipe_name ) ;
if ( state - > ep . name = = NULL ) {
DEBUG ( 0 , ( " Out of memory \n " ) ) ;
goto out ;
}
state - > fd = create_named_pipe_socket ( pipe_name ) ;
if ( state - > fd = = - 1 ) {
goto out ;
}
2011-08-29 09:49:22 +02:00
rc = listen ( state - > fd , 5 ) ;
if ( rc < 0 ) {
DEBUG ( 0 , ( " Failed to listen on pipe socket %s: %s \n " ,
pipe_name , strerror ( errno ) ) ) ;
goto out ;
}
2011-04-21 15:05:12 -04:00
state - > ev_ctx = ev_ctx ;
state - > msg_ctx = msg_ctx ;
2010-06-18 11:00:38 -04:00
2017-02-18 09:02:23 +13:00
DEBUG ( 10 , ( " Opened pipe socket fd %d for %s \n " ,
2010-06-18 11:00:38 -04:00
state - > fd , pipe_name ) ) ;
fde = tevent_add_fd ( ev_ctx ,
state , state - > fd , TEVENT_FD_READ ,
named_pipe_listener , state ) ;
if ( ! fde ) {
DEBUG ( 0 , ( " Failed to add event handler! \n " ) ) ;
goto out ;
}
tevent_fd_set_auto_close ( fde ) ;
return true ;
out :
if ( state - > fd ! = - 1 ) {
close ( state - > fd ) ;
}
TALLOC_FREE ( state ) ;
return false ;
}
2010-05-20 09:16:29 -04:00
static void named_pipe_listener ( struct tevent_context * ev ,
struct tevent_fd * fde ,
uint16_t flags ,
void * private_data )
{
2011-02-15 12:57:26 +01:00
struct dcerpc_ncacn_listen_state * state =
2010-05-20 09:16:29 -04:00
talloc_get_type_abort ( private_data ,
2011-02-15 12:57:26 +01:00
struct dcerpc_ncacn_listen_state ) ;
2010-05-20 09:16:29 -04:00
struct sockaddr_un sunaddr ;
socklen_t len ;
int sd = - 1 ;
/* TODO: should we have a limit to the number of clients ? */
len = sizeof ( sunaddr ) ;
2011-03-03 15:06:45 +01:00
sd = accept ( state - > fd ,
( struct sockaddr * ) ( void * ) & sunaddr , & len ) ;
2010-05-20 09:16:29 -04:00
if ( sd = = - 1 ) {
2011-03-03 15:06:45 +01:00
if ( errno ! = EINTR ) {
DEBUG ( 6 , ( " Failed to get a valid socket [%s] \n " ,
strerror ( errno ) ) ) ;
}
2010-05-20 09:16:29 -04:00
return ;
}
2017-12-11 09:39:43 +13:00
smb_set_close_on_exec ( sd ) ;
2010-05-20 09:16:29 -04:00
DEBUG ( 6 , ( " Accepted socket %d \n " , sd ) ) ;
2011-07-08 11:38:03 +02:00
named_pipe_accept_function ( state - > ev_ctx ,
state - > msg_ctx ,
state - > ep . name ,
2011-04-25 17:24:15 -04:00
sd , NULL , 0 ) ;
2010-05-20 09:16:29 -04:00
}
2010-06-18 10:56:58 -04:00
/* This is the core of the rpc server.
* Accepts connections from clients and process requests using the appropriate
* dispatcher table . */
2011-04-25 17:24:15 -04:00
static int named_pipe_destructor ( struct named_pipe_client * npc )
{
if ( npc - > term_fn ) {
npc - > term_fn ( npc - > private_data ) ;
}
return 0 ;
}
2013-09-25 11:34:56 +02:00
struct named_pipe_client * named_pipe_client_init ( TALLOC_CTX * mem_ctx ,
struct tevent_context * ev_ctx ,
struct messaging_context * msg_ctx ,
const char * pipe_name ,
named_pipe_termination_fn * term_fn ,
uint16_t file_type ,
uint16_t device_state ,
uint64_t allocation_size ,
void * private_data )
{
struct named_pipe_client * npc ;
npc = talloc_zero ( mem_ctx , struct named_pipe_client ) ;
if ( npc = = NULL ) {
DEBUG ( 0 , ( " Out of memory! \n " ) ) ;
return NULL ;
}
talloc_set_destructor ( npc , named_pipe_destructor ) ;
npc - > pipe_name = talloc_strdup ( npc , pipe_name ) ;
if ( npc - > pipe_name = = NULL ) {
DEBUG ( 0 , ( " Out of memory! \n " ) ) ;
talloc_free ( npc ) ;
return NULL ;
}
npc - > ev = ev_ctx ;
npc - > msg_ctx = msg_ctx ;
npc - > term_fn = term_fn ;
npc - > private_data = private_data ;
npc - > file_type = file_type ;
npc - > device_state = device_state ;
npc - > allocation_size = allocation_size ;
return npc ;
}
2010-06-18 10:56:58 -04:00
static void named_pipe_accept_done ( struct tevent_req * subreq ) ;
2011-04-21 15:05:12 -04:00
void named_pipe_accept_function ( struct tevent_context * ev_ctx ,
struct messaging_context * msg_ctx ,
2011-04-25 17:24:15 -04:00
const char * pipe_name , int fd ,
named_pipe_termination_fn * term_fn ,
void * private_data )
2010-05-20 09:16:29 -04:00
{
2010-06-18 10:56:58 -04:00
struct named_pipe_client * npc ;
struct tstream_context * plain ;
struct tevent_req * subreq ;
int ret ;
2011-07-26 12:07:20 +02:00
npc = talloc_zero ( ev_ctx , struct named_pipe_client ) ;
2010-06-18 10:56:58 -04:00
if ( ! npc ) {
DEBUG ( 0 , ( " Out of memory! \n " ) ) ;
close ( fd ) ;
return ;
}
2011-07-21 11:21:30 +02:00
npc - > pipe_name = talloc_strdup ( npc , pipe_name ) ;
if ( npc - > pipe_name = = NULL ) {
DEBUG ( 0 , ( " Out of memory! \n " ) ) ;
TALLOC_FREE ( npc ) ;
close ( fd ) ;
return ;
}
2011-07-08 11:38:03 +02:00
npc - > ev = ev_ctx ;
npc - > msg_ctx = msg_ctx ;
2011-04-25 17:24:15 -04:00
npc - > term_fn = term_fn ;
npc - > private_data = private_data ;
talloc_set_destructor ( npc , named_pipe_destructor ) ;
2010-06-18 10:56:58 -04:00
/* make sure socket is in NON blocking state */
ret = set_blocking ( fd , false ) ;
if ( ret ! = 0 ) {
DEBUG ( 2 , ( " Failed to make socket non-blocking \n " ) ) ;
TALLOC_FREE ( npc ) ;
close ( fd ) ;
return ;
}
ret = tstream_bsd_existing_socket ( npc , fd , & plain ) ;
if ( ret ! = 0 ) {
DEBUG ( 2 , ( " Failed to create tstream socket \n " ) ) ;
TALLOC_FREE ( npc ) ;
close ( fd ) ;
return ;
}
npc - > file_type = FILE_TYPE_MESSAGE_MODE_PIPE ;
npc - > device_state = 0xff | 0x0400 | 0x0100 ;
npc - > allocation_size = 4096 ;
subreq = tstream_npa_accept_existing_send ( npc , npc - > ev , plain ,
npc - > file_type ,
npc - > device_state ,
npc - > allocation_size ) ;
if ( ! subreq ) {
DEBUG ( 2 , ( " Failed to start async accept procedure \n " ) ) ;
TALLOC_FREE ( npc ) ;
close ( fd ) ;
return ;
}
tevent_req_set_callback ( subreq , named_pipe_accept_done , npc ) ;
}
static void named_pipe_packet_done ( struct tevent_req * subreq ) ;
static void named_pipe_accept_done ( struct tevent_req * subreq )
{
2011-04-05 16:15:27 +10:00
struct auth_session_info_transport * session_info_transport ;
2010-06-18 10:56:58 -04:00
struct named_pipe_client * npc =
tevent_req_callback_data ( subreq , struct named_pipe_client ) ;
int error ;
int ret ;
ret = tstream_npa_accept_existing_recv ( subreq , & error , npc ,
& npc - > tstream ,
2017-03-10 12:33:06 +13:00
& npc - > remote_client_addr ,
& npc - > remote_client_name ,
& npc - > local_server_addr ,
& npc - > local_server_name ,
2011-04-05 16:15:27 +10:00
& session_info_transport ) ;
npc - > session_info = talloc_move ( npc , & session_info_transport - > session_info ) ;
2010-06-18 10:56:58 -04:00
TALLOC_FREE ( subreq ) ;
if ( ret ! = 0 ) {
DEBUG ( 2 , ( " Failed to accept named pipe connection! (%s) \n " ,
strerror ( error ) ) ) ;
TALLOC_FREE ( npc ) ;
return ;
}
ret = make_server_pipes_struct ( npc ,
2011-07-21 10:43:56 +02:00
npc - > msg_ctx ,
2011-05-25 17:26:01 +02:00
npc - > pipe_name , NCACN_NP ,
2017-03-10 12:33:06 +13:00
npc - > remote_client_addr ,
2017-03-10 12:38:33 +13:00
npc - > local_server_addr ,
2014-04-17 14:25:48 +02:00
npc - > session_info ,
& npc - > p , & error ) ;
2010-06-18 10:56:58 -04:00
if ( ret ! = 0 ) {
DEBUG ( 2 , ( " Failed to create pipes_struct! (%s) \n " ,
strerror ( error ) ) ) ;
goto fail ;
}
npc - > write_queue = tevent_queue_create ( npc , " np_server_write_queue " ) ;
if ( ! npc - > write_queue ) {
DEBUG ( 2 , ( " Failed to set up write queue! \n " ) ) ;
goto fail ;
}
2011-03-31 12:02:26 +02:00
/* And now start receiving and processing packets */
2011-02-25 13:50:37 +01:00
subreq = dcerpc_read_ncacn_packet_send ( npc , npc - > ev , npc - > tstream ) ;
2010-06-18 10:56:58 -04:00
if ( ! subreq ) {
2018-05-04 22:23:01 +02:00
DEBUG ( 2 , ( " Failed to start receiving packets \n " ) ) ;
2010-06-18 10:56:58 -04:00
goto fail ;
}
tevent_req_set_callback ( subreq , named_pipe_packet_process , npc ) ;
return ;
fail :
DEBUG ( 2 , ( " Fatal error. Terminating client(%s) connection! \n " ,
2017-03-10 12:33:06 +13:00
npc - > remote_client_name ) ) ;
2010-06-18 10:56:58 -04:00
/* terminate client connection */
talloc_free ( npc ) ;
return ;
}
2013-09-25 10:25:39 +02:00
void named_pipe_packet_process ( struct tevent_req * subreq )
2010-06-18 10:56:58 -04:00
{
struct named_pipe_client * npc =
tevent_req_callback_data ( subreq , struct named_pipe_client ) ;
struct _output_data * out = & npc - > p - > out_data ;
2010-09-15 13:24:44 +02:00
DATA_BLOB recv_buffer = data_blob_null ;
2011-02-25 13:50:37 +01:00
struct ncacn_packet * pkt ;
2010-06-18 10:56:58 -04:00
NTSTATUS status ;
uint32_t to_send ;
2011-08-03 23:44:21 +02:00
size_t i ;
2010-06-18 10:56:58 -04:00
bool ok ;
2011-02-25 13:50:37 +01:00
status = dcerpc_read_ncacn_packet_recv ( subreq , npc , & pkt , & recv_buffer ) ;
2010-06-18 10:56:58 -04:00
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto fail ;
}
2013-12-10 13:59:06 +01:00
/* dcerpc_read_ncacn_packet_recv() returns a full PDU */
npc - > p - > in_data . pdu_needed_len = 0 ;
npc - > p - > in_data . pdu = recv_buffer ;
if ( dcerpc_get_endian_flag ( & recv_buffer ) & DCERPC_DREP_LE ) {
npc - > p - > endian = RPC_LITTLE_ENDIAN ;
} else {
npc - > p - > endian = RPC_BIG_ENDIAN ;
2010-06-18 10:56:58 -04:00
}
2013-12-10 13:59:06 +01:00
DEBUG ( 10 , ( " PDU is in %s Endian format! \n " ,
npc - > p - > endian ? " Big " : " Little " ) ) ;
process_complete_pdu ( npc - > p , pkt ) ;
2010-06-18 10:56:58 -04:00
2013-12-10 13:59:06 +01:00
/* reset pipe state and free PDU */
npc - > p - > in_data . pdu . length = 0 ;
2010-06-18 10:56:58 -04:00
talloc_free ( recv_buffer . data ) ;
2011-02-25 13:50:37 +01:00
talloc_free ( pkt ) ;
2010-06-18 10:56:58 -04:00
/* this is needed because of the way DCERPC Binds work in
* the RPC marshalling code */
to_send = out - > frag . length - out - > current_pdu_sent ;
if ( to_send > 0 ) {
npc - > iov = talloc_zero ( npc , struct iovec ) ;
if ( ! npc - > iov ) {
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
npc - > count = 1 ;
npc - > iov [ 0 ] . iov_base = out - > frag . data
+ out - > current_pdu_sent ;
npc - > iov [ 0 ] . iov_len = to_send ;
out - > current_pdu_sent + = to_send ;
}
/* this condition is false for bind packets, or when we haven't
* yet got a full request , and need to wait for more data from
* the client */
while ( out - > data_sent_length < out - > rdata . length ) {
ok = create_next_pdu ( npc - > p ) ;
if ( ! ok ) {
DEBUG ( 3 , ( " Failed to create next PDU! \n " ) ) ;
status = NT_STATUS_UNEXPECTED_IO_ERROR ;
goto fail ;
}
npc - > iov = talloc_realloc ( npc , npc - > iov ,
struct iovec , npc - > count + 1 ) ;
if ( ! npc - > iov ) {
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
npc - > iov [ npc - > count ] . iov_base = out - > frag . data ;
npc - > iov [ npc - > count ] . iov_len = out - > frag . length ;
npc - > count + + ;
}
/* we still don't have a complete request, go back and wait for more
* data */
if ( npc - > count = = 0 ) {
/* Wait for the next packet */
2011-02-25 13:50:37 +01:00
subreq = dcerpc_read_ncacn_packet_send ( npc , npc - > ev , npc - > tstream ) ;
2010-06-18 10:56:58 -04:00
if ( ! subreq ) {
2018-05-04 22:23:01 +02:00
DEBUG ( 2 , ( " Failed to start receiving packets \n " ) ) ;
2010-06-18 10:56:58 -04:00
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
tevent_req_set_callback ( subreq , named_pipe_packet_process , npc ) ;
return ;
}
2011-08-03 23:44:21 +02:00
DEBUG ( 10 , ( " Sending %u fragments in a total of %u bytes \n " ,
( unsigned int ) npc - > count ,
2010-06-18 10:56:58 -04:00
( unsigned int ) npc - > p - > out_data . data_sent_length ) ) ;
2011-08-03 23:44:21 +02:00
for ( i = 0 ; i < npc - > count ; i + + ) {
DEBUG ( 10 , ( " Sending PDU number: %d, PDU Length: %u \n " ,
( unsigned int ) i ,
( unsigned int ) npc - > iov [ i ] . iov_len ) ) ;
dump_data ( 11 , ( const uint8_t * ) npc - > iov [ i ] . iov_base ,
npc - > iov [ i ] . iov_len ) ;
subreq = tstream_writev_queue_send ( npc ,
npc - > ev ,
npc - > tstream ,
npc - > write_queue ,
( npc - > iov + i ) ,
1 ) ;
if ( ! subreq ) {
DEBUG ( 2 , ( " Failed to send packet \n " ) ) ;
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
tevent_req_set_callback ( subreq , named_pipe_packet_done , npc ) ;
2010-06-18 10:56:58 -04:00
}
2011-08-03 23:44:21 +02:00
2010-06-18 10:56:58 -04:00
return ;
fail :
DEBUG ( 2 , ( " Fatal error(%s). "
" Terminating client(%s) connection! \n " ,
2017-03-10 12:33:06 +13:00
nt_errstr ( status ) , npc - > remote_client_name ) ) ;
2010-06-18 10:56:58 -04:00
/* terminate client connection */
talloc_free ( npc ) ;
2010-05-20 09:16:29 -04:00
return ;
}
2010-06-18 10:56:58 -04:00
static void named_pipe_packet_done ( struct tevent_req * subreq )
{
struct named_pipe_client * npc =
tevent_req_callback_data ( subreq , struct named_pipe_client ) ;
int sys_errno ;
int ret ;
ret = tstream_writev_queue_recv ( subreq , & sys_errno ) ;
TALLOC_FREE ( subreq ) ;
if ( ret = = - 1 ) {
DEBUG ( 2 , ( " Writev failed! \n " ) ) ;
goto fail ;
}
2011-08-03 23:44:21 +02:00
if ( tevent_queue_length ( npc - > write_queue ) > 0 ) {
return ;
}
2015-12-23 12:40:58 +01:00
if ( npc - > p - > fault_state ! = 0 ) {
DEBUG ( 2 , ( " Disconnect after fault \n " ) ) ;
sys_errno = EINVAL ;
goto fail ;
}
2010-06-18 10:56:58 -04:00
/* clear out any data that may have been left around */
npc - > count = 0 ;
TALLOC_FREE ( npc - > iov ) ;
data_blob_free ( & npc - > p - > in_data . data ) ;
data_blob_free ( & npc - > p - > out_data . frag ) ;
data_blob_free ( & npc - > p - > out_data . rdata ) ;
2011-07-26 12:07:20 +02:00
talloc_free_children ( npc - > p - > mem_ctx ) ;
2010-06-18 10:56:58 -04:00
/* Wait for the next packet */
2011-02-25 13:50:37 +01:00
subreq = dcerpc_read_ncacn_packet_send ( npc , npc - > ev , npc - > tstream ) ;
2010-06-18 10:56:58 -04:00
if ( ! subreq ) {
2018-05-04 22:23:01 +02:00
DEBUG ( 2 , ( " Failed to start receiving packets \n " ) ) ;
2010-06-18 10:56:58 -04:00
sys_errno = ENOMEM ;
goto fail ;
}
tevent_req_set_callback ( subreq , named_pipe_packet_process , npc ) ;
return ;
fail :
DEBUG ( 2 , ( " Fatal error(%s). "
" Terminating client(%s) connection! \n " ,
2017-03-10 12:33:06 +13:00
strerror ( sys_errno ) , npc - > remote_client_name ) ) ;
2010-06-18 10:56:58 -04:00
/* terminate client connection */
talloc_free ( npc ) ;
return ;
2011-02-22 20:40:24 -05:00
}
2011-02-14 10:29:14 +01:00
/********************************************************************
* Start listening on the tcp / ip socket
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void dcerpc_ncacn_tcpip_listener ( struct tevent_context * ev ,
struct tevent_fd * fde ,
uint16_t flags ,
void * private_data ) ;
2011-05-23 16:00:30 +02:00
int create_tcpip_socket ( const struct sockaddr_storage * ifss , uint16_t * port )
{
int fd = - 1 ;
if ( * port = = 0 ) {
uint16_t i ;
2017-01-16 12:05:09 +01:00
for ( i = lp_rpc_low_port ( ) ; i < = lp_rpc_high_port ( ) ; i + + ) {
2011-05-23 16:00:30 +02:00
fd = open_socket_in ( SOCK_STREAM ,
i ,
0 ,
ifss ,
false ) ;
2013-08-18 20:41:51 +00:00
if ( fd > = 0 ) {
2011-05-23 16:00:30 +02:00
* port = i ;
break ;
}
}
} else {
fd = open_socket_in ( SOCK_STREAM ,
* port ,
0 ,
ifss ,
true ) ;
}
if ( fd = = - 1 ) {
DEBUG ( 0 , ( " Failed to create socket on port %u! \n " , * port ) ) ;
return - 1 ;
}
DEBUG ( 10 , ( " Opened tcpip socket fd %d for port %u \n " , fd , * port ) ) ;
return fd ;
}
2011-02-14 10:29:14 +01:00
uint16_t setup_dcerpc_ncacn_tcpip_socket ( struct tevent_context * ev_ctx ,
struct messaging_context * msg_ctx ,
const struct sockaddr_storage * ifss ,
uint16_t port )
{
struct dcerpc_ncacn_listen_state * state ;
struct tevent_fd * fde ;
int rc ;
state = talloc ( ev_ctx , struct dcerpc_ncacn_listen_state ) ;
if ( state = = NULL ) {
DEBUG ( 0 , ( " setup_dcerpc_ncacn_tcpip_socket: Out of memory \n " ) ) ;
2011-02-22 20:40:24 -05:00
return 0 ;
2011-02-14 10:29:14 +01:00
}
state - > fd = - 1 ;
state - > ep . port = port ;
2011-03-14 12:29:49 +01:00
state - > disconnect_fn = NULL ;
2011-02-14 10:29:14 +01:00
2011-05-23 16:00:30 +02:00
state - > fd = create_tcpip_socket ( ifss , & state - > ep . port ) ;
2011-02-14 10:29:14 +01:00
if ( state - > fd = = - 1 ) {
goto out ;
}
state - > ev_ctx = ev_ctx ;
state - > msg_ctx = msg_ctx ;
/* ready to listen */
set_socket_options ( state - > fd , " SO_KEEPALIVE " ) ;
set_socket_options ( state - > fd , lp_socket_options ( ) ) ;
/* Set server socket to non-blocking for the accept. */
set_blocking ( state - > fd , false ) ;
rc = listen ( state - > fd , SMBD_LISTEN_BACKLOG ) ;
if ( rc = = - 1 ) {
DEBUG ( 0 , ( " setup_tcpip_socket: listen - %s \n " , strerror ( errno ) ) ) ;
goto out ;
}
2018-05-04 22:21:53 +02:00
DEBUG ( 10 , ( " setup_tcpip_socket: opened socket fd %d for port %u \n " ,
2011-02-14 10:29:14 +01:00
state - > fd , state - > ep . port ) ) ;
fde = tevent_add_fd ( state - > ev_ctx ,
state ,
state - > fd ,
TEVENT_FD_READ ,
dcerpc_ncacn_tcpip_listener ,
state ) ;
if ( fde = = NULL ) {
DEBUG ( 0 , ( " setup_tcpip_socket: Failed to add event handler! \n " ) ) ;
goto out ;
}
tevent_fd_set_auto_close ( fde ) ;
return state - > ep . port ;
out :
if ( state - > fd ! = - 1 ) {
close ( state - > fd ) ;
}
TALLOC_FREE ( state ) ;
return 0 ;
}
static void dcerpc_ncacn_tcpip_listener ( struct tevent_context * ev ,
struct tevent_fd * fde ,
uint16_t flags ,
void * private_data )
{
struct dcerpc_ncacn_listen_state * state =
talloc_get_type_abort ( private_data ,
struct dcerpc_ncacn_listen_state ) ;
2011-02-22 15:14:34 +01:00
struct tsocket_address * cli_addr = NULL ;
struct tsocket_address * srv_addr = NULL ;
2011-02-14 10:29:14 +01:00
struct sockaddr_storage addr ;
socklen_t in_addrlen = sizeof ( addr ) ;
int s = - 1 ;
int rc ;
2011-03-03 15:07:02 +01:00
s = accept ( state - > fd , ( struct sockaddr * ) ( void * ) & addr , & in_addrlen ) ;
2011-02-14 10:29:14 +01:00
if ( s = = - 1 ) {
2011-03-03 15:07:02 +01:00
if ( errno ! = EINTR ) {
DEBUG ( 0 , ( " tcpip_listener accept: %s \n " ,
strerror ( errno ) ) ) ;
}
2011-02-14 10:29:14 +01:00
return ;
}
2017-12-11 09:39:43 +13:00
smb_set_close_on_exec ( s ) ;
2011-02-14 10:29:14 +01:00
rc = tsocket_address_bsd_from_sockaddr ( state ,
( struct sockaddr * ) ( void * ) & addr ,
in_addrlen ,
& cli_addr ) ;
if ( rc < 0 ) {
close ( s ) ;
return ;
}
2011-02-22 15:14:34 +01:00
rc = getsockname ( s , ( struct sockaddr * ) ( void * ) & addr , & in_addrlen ) ;
if ( rc < 0 ) {
close ( s ) ;
return ;
}
rc = tsocket_address_bsd_from_sockaddr ( state ,
( struct sockaddr * ) ( void * ) & addr ,
in_addrlen ,
& srv_addr ) ;
if ( rc < 0 ) {
close ( s ) ;
return ;
}
2011-02-14 10:29:14 +01:00
DEBUG ( 6 , ( " tcpip_listener: Accepted socket %d \n " , s ) ) ;
dcerpc_ncacn_accept ( state - > ev_ctx ,
state - > msg_ctx ,
NCACN_IP_TCP ,
NULL ,
cli_addr ,
2011-02-22 15:14:34 +01:00
srv_addr ,
2011-03-14 12:29:49 +01:00
s ,
NULL ) ;
2011-02-14 10:29:14 +01:00
}
2011-02-22 20:40:54 -05:00
/********************************************************************
* Start listening on the ncalrpc socket
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void dcerpc_ncalrpc_listener ( struct tevent_context * ev ,
struct tevent_fd * fde ,
uint16_t flags ,
void * private_data ) ;
2011-08-18 19:05:10 +02:00
int create_dcerpc_ncalrpc_socket ( const char * name )
{
int fd = - 1 ;
if ( name = = NULL ) {
name = " DEFAULT " ;
}
2014-07-27 19:18:09 +02:00
if ( ! directory_create_or_exist ( lp_ncalrpc_dir ( ) , 0755 ) ) {
2011-08-18 19:05:10 +02:00
DEBUG ( 0 , ( " Failed to create ncalrpc directory %s - %s \n " ,
lp_ncalrpc_dir ( ) , strerror ( errno ) ) ) ;
return - 1 ;
}
fd = create_pipe_sock ( lp_ncalrpc_dir ( ) , name , 0755 ) ;
if ( fd = = - 1 ) {
DEBUG ( 0 , ( " Failed to create ncalrpc socket! [%s/%s] \n " ,
lp_ncalrpc_dir ( ) , name ) ) ;
return - 1 ;
}
2017-02-18 09:02:23 +13:00
DEBUG ( 10 , ( " Opened ncalrpc socket fd %d for %s \n " , fd , name ) ) ;
2011-08-18 19:05:10 +02:00
return fd ;
}
2011-02-22 20:40:54 -05:00
bool setup_dcerpc_ncalrpc_socket ( struct tevent_context * ev_ctx ,
struct messaging_context * msg_ctx ,
2011-03-14 12:29:49 +01:00
const char * name ,
dcerpc_ncacn_disconnect_fn fn )
2011-02-22 20:40:54 -05:00
{
struct dcerpc_ncacn_listen_state * state ;
struct tevent_fd * fde ;
2011-08-29 09:49:22 +02:00
int rc ;
2011-02-22 20:40:54 -05:00
state = talloc ( ev_ctx , struct dcerpc_ncacn_listen_state ) ;
if ( state = = NULL ) {
DEBUG ( 0 , ( " Out of memory \n " ) ) ;
return false ;
}
state - > fd = - 1 ;
2011-03-14 12:29:49 +01:00
state - > disconnect_fn = fn ;
2011-02-22 20:40:54 -05:00
if ( name = = NULL ) {
name = " DEFAULT " ;
}
2011-08-18 19:05:10 +02:00
state - > ep . name = talloc_strdup ( state , name ) ;
2011-02-22 20:40:54 -05:00
if ( state - > ep . name = = NULL ) {
DEBUG ( 0 , ( " Out of memory \n " ) ) ;
talloc_free ( state ) ;
return false ;
}
2011-08-18 19:05:10 +02:00
state - > fd = create_dcerpc_ncalrpc_socket ( name ) ;
2011-02-22 20:40:54 -05:00
if ( state - > fd = = - 1 ) {
goto out ;
}
2011-08-29 09:49:22 +02:00
rc = listen ( state - > fd , 5 ) ;
if ( rc < 0 ) {
DEBUG ( 0 , ( " Failed to listen on ncalrpc socket %s: %s \n " ,
name , strerror ( errno ) ) ) ;
goto out ;
}
2011-02-22 20:40:54 -05:00
state - > ev_ctx = ev_ctx ;
state - > msg_ctx = msg_ctx ;
/* Set server socket to non-blocking for the accept. */
set_blocking ( state - > fd , false ) ;
fde = tevent_add_fd ( state - > ev_ctx ,
state ,
state - > fd ,
TEVENT_FD_READ ,
dcerpc_ncalrpc_listener ,
state ) ;
if ( fde = = NULL ) {
DEBUG ( 0 , ( " Failed to add event handler for ncalrpc! \n " ) ) ;
goto out ;
}
tevent_fd_set_auto_close ( fde ) ;
return true ;
out :
if ( state - > fd ! = - 1 ) {
close ( state - > fd ) ;
}
TALLOC_FREE ( state ) ;
return 0 ;
}
static void dcerpc_ncalrpc_listener ( struct tevent_context * ev ,
struct tevent_fd * fde ,
uint16_t flags ,
void * private_data )
{
struct dcerpc_ncacn_listen_state * state =
talloc_get_type_abort ( private_data ,
struct dcerpc_ncacn_listen_state ) ;
2017-02-24 13:29:12 +13:00
struct tsocket_address * cli_addr = NULL , * srv_addr = NULL ;
2011-02-22 20:40:54 -05:00
struct sockaddr_un sunaddr ;
struct sockaddr * addr = ( struct sockaddr * ) ( void * ) & sunaddr ;
2011-02-25 14:26:29 +01:00
socklen_t len = sizeof ( sunaddr ) ;
2017-02-24 13:29:12 +13:00
struct sockaddr_un sunaddr_server ;
struct sockaddr * addr_server = ( struct sockaddr * ) ( void * ) & sunaddr_server ;
socklen_t len_server = sizeof ( sunaddr_server ) ;
2011-02-22 20:40:54 -05:00
int sd = - 1 ;
int rc ;
2011-02-25 14:26:29 +01:00
ZERO_STRUCT ( sunaddr ) ;
2017-02-24 13:29:12 +13:00
ZERO_STRUCT ( sunaddr_server ) ;
2011-02-25 14:26:29 +01:00
2011-03-03 15:07:16 +01:00
sd = accept ( state - > fd , addr , & len ) ;
2011-02-22 20:40:54 -05:00
if ( sd = = - 1 ) {
2011-03-03 15:07:16 +01:00
if ( errno ! = EINTR ) {
DEBUG ( 0 , ( " ncalrpc accept() failed: %s \n " , strerror ( errno ) ) ) ;
}
2011-02-22 20:40:54 -05:00
return ;
}
2017-12-11 09:39:43 +13:00
smb_set_close_on_exec ( sd ) ;
2011-02-22 20:40:54 -05:00
rc = tsocket_address_bsd_from_sockaddr ( state ,
addr , len ,
& cli_addr ) ;
if ( rc < 0 ) {
close ( sd ) ;
return ;
}
2017-02-24 13:29:12 +13:00
rc = getsockname ( sd , addr_server , & len_server ) ;
if ( rc < 0 ) {
close ( sd ) ;
return ;
}
rc = tsocket_address_bsd_from_sockaddr ( state ,
addr_server ,
len_server ,
& srv_addr ) ;
if ( rc < 0 ) {
close ( sd ) ;
return ;
}
DEBUG ( 10 , ( " Accepted ncalrpc socket %s (fd: %d) \n " ,
sunaddr . sun_path , sd ) ) ;
2011-02-22 20:40:54 -05:00
dcerpc_ncacn_accept ( state - > ev_ctx ,
state - > msg_ctx ,
2011-05-25 17:26:01 +02:00
NCALRPC ,
2011-07-15 18:03:47 +02:00
state - > ep . name ,
2017-02-24 13:29:12 +13:00
cli_addr , srv_addr , sd ,
2011-03-14 12:29:49 +01:00
state - > disconnect_fn ) ;
2011-02-22 20:40:54 -05:00
}
2011-02-14 10:29:14 +01:00
struct dcerpc_ncacn_conn {
enum dcerpc_transport_t transport ;
int sock ;
struct pipes_struct * p ;
2011-03-14 12:29:49 +01:00
dcerpc_ncacn_disconnect_fn disconnect_fn ;
2011-02-14 10:29:14 +01:00
struct tevent_context * ev_ctx ;
struct messaging_context * msg_ctx ;
struct tstream_context * tstream ;
struct tevent_queue * send_queue ;
2017-03-10 12:13:24 +13:00
struct tsocket_address * remote_client_addr ;
char * remote_client_name ;
struct tsocket_address * local_server_addr ;
char * local_server_name ;
2011-04-05 16:15:27 +10:00
struct auth_session_info * session_info ;
2011-02-14 10:29:14 +01:00
struct iovec * iov ;
size_t count ;
} ;
static void dcerpc_ncacn_packet_process ( struct tevent_req * subreq ) ;
static void dcerpc_ncacn_packet_done ( struct tevent_req * subreq ) ;
2011-07-20 16:26:48 +02:00
void dcerpc_ncacn_accept ( struct tevent_context * ev_ctx ,
struct messaging_context * msg_ctx ,
enum dcerpc_transport_t transport ,
const char * name ,
struct tsocket_address * cli_addr ,
struct tsocket_address * srv_addr ,
int s ,
dcerpc_ncacn_disconnect_fn fn ) {
2011-02-14 10:29:14 +01:00
struct dcerpc_ncacn_conn * ncacn_conn ;
struct tevent_req * subreq ;
char * pipe_name ;
NTSTATUS status ;
int sys_errno ;
2011-03-02 10:56:46 +01:00
uid_t uid ;
2012-03-24 16:00:36 +01:00
gid_t gid ;
2011-02-14 10:29:14 +01:00
int rc ;
2011-02-22 20:40:24 -05:00
DEBUG ( 10 , ( " dcerpc_ncacn_accept \n " ) ) ;
2011-02-14 10:29:14 +01:00
ncacn_conn = talloc_zero ( ev_ctx , struct dcerpc_ncacn_conn ) ;
if ( ncacn_conn = = NULL ) {
2011-02-22 20:40:24 -05:00
DEBUG ( 0 , ( " Out of memory! \n " ) ) ;
2011-02-14 10:29:14 +01:00
close ( s ) ;
return ;
}
ncacn_conn - > transport = transport ;
ncacn_conn - > ev_ctx = ev_ctx ;
ncacn_conn - > msg_ctx = msg_ctx ;
ncacn_conn - > sock = s ;
2011-03-14 12:29:49 +01:00
ncacn_conn - > disconnect_fn = fn ;
2011-02-14 10:29:14 +01:00
2017-03-10 12:13:24 +13:00
ncacn_conn - > remote_client_addr = talloc_move ( ncacn_conn , & cli_addr ) ;
if ( tsocket_address_is_inet ( ncacn_conn - > remote_client_addr , " ip " ) ) {
ncacn_conn - > remote_client_name =
tsocket_address_inet_addr_string ( ncacn_conn - > remote_client_addr ,
2011-02-22 15:14:34 +01:00
ncacn_conn ) ;
} else {
2017-03-10 12:13:24 +13:00
ncacn_conn - > remote_client_name =
tsocket_address_unix_path ( ncacn_conn - > remote_client_addr ,
2011-02-22 15:14:34 +01:00
ncacn_conn ) ;
}
2017-03-10 12:13:24 +13:00
if ( ncacn_conn - > remote_client_name = = NULL ) {
2017-02-24 13:29:12 +13:00
DEBUG ( 0 , ( " Out of memory obtaining remote socket address as a string! \n " ) ) ;
2011-02-22 15:14:34 +01:00
talloc_free ( ncacn_conn ) ;
close ( s ) ;
return ;
}
if ( srv_addr ! = NULL ) {
2017-03-10 12:13:24 +13:00
ncacn_conn - > local_server_addr = talloc_move ( ncacn_conn , & srv_addr ) ;
2011-02-22 15:14:34 +01:00
2017-03-10 12:13:24 +13:00
if ( tsocket_address_is_inet ( ncacn_conn - > local_server_addr , " ip " ) ) {
ncacn_conn - > local_server_name =
tsocket_address_inet_addr_string ( ncacn_conn - > local_server_addr ,
2017-02-24 13:29:12 +13:00
ncacn_conn ) ;
} else {
2017-03-10 12:13:24 +13:00
ncacn_conn - > local_server_name =
tsocket_address_unix_path ( ncacn_conn - > local_server_addr ,
2017-02-24 13:29:12 +13:00
ncacn_conn ) ;
}
2017-03-10 12:13:24 +13:00
if ( ncacn_conn - > local_server_name = = NULL ) {
2017-02-24 13:29:12 +13:00
DEBUG ( 0 , ( " Out of memory obtaining local socket address as a string! \n " ) ) ;
2011-02-22 15:14:34 +01:00
talloc_free ( ncacn_conn ) ;
close ( s ) ;
return ;
}
}
2011-02-14 10:29:14 +01:00
2011-03-01 19:10:02 +01:00
switch ( transport ) {
case NCACN_IP_TCP :
2017-03-10 12:13:24 +13:00
pipe_name = tsocket_address_string ( ncacn_conn - > remote_client_addr ,
2011-03-01 19:10:02 +01:00
ncacn_conn ) ;
if ( pipe_name = = NULL ) {
close ( s ) ;
talloc_free ( ncacn_conn ) ;
return ;
}
break ;
case NCALRPC :
2012-03-24 16:00:36 +01:00
rc = getpeereid ( s , & uid , & gid ) ;
2011-03-02 10:56:46 +01:00
if ( rc < 0 ) {
2011-09-15 17:25:52 +02:00
DEBUG ( 2 , ( " Failed to get ncalrpc connecting "
" uid - %s! \n " , strerror ( errno ) ) ) ;
2011-03-02 10:56:46 +01:00
} else {
if ( uid = = sec_initial_uid ( ) ) {
2017-03-10 12:13:24 +13:00
TALLOC_FREE ( ncacn_conn - > remote_client_addr ) ;
2014-04-17 11:00:54 +02:00
rc = tsocket_address_unix_from_path ( ncacn_conn ,
2017-07-24 11:00:45 +12:00
AS_SYSTEM_MAGIC_PATH_TOKEN ,
2017-03-10 12:13:24 +13:00
& ncacn_conn - > remote_client_addr ) ;
2014-04-17 11:00:54 +02:00
if ( rc < 0 ) {
2017-02-24 13:29:12 +13:00
DEBUG ( 0 , ( " Out of memory building magic ncalrpc_as_system path! \n " ) ) ;
2014-04-17 11:00:54 +02:00
talloc_free ( ncacn_conn ) ;
close ( s ) ;
return ;
}
2017-03-10 12:13:24 +13:00
TALLOC_FREE ( ncacn_conn - > remote_client_name ) ;
ncacn_conn - > remote_client_name
= tsocket_address_unix_path ( ncacn_conn - > remote_client_addr ,
ncacn_conn ) ;
if ( ncacn_conn - > remote_client_name = = NULL ) {
2017-02-24 13:29:12 +13:00
DEBUG ( 0 , ( " Out of memory getting magic ncalrpc_as_system string! \n " ) ) ;
2014-04-17 11:00:54 +02:00
talloc_free ( ncacn_conn ) ;
close ( s ) ;
return ;
}
2011-03-02 10:56:46 +01:00
}
}
2017-07-26 17:42:46 +02:00
FALL_THROUGH ;
2011-03-01 19:10:02 +01:00
case NCACN_NP :
pipe_name = talloc_strdup ( ncacn_conn ,
name ) ;
if ( pipe_name = = NULL ) {
close ( s ) ;
talloc_free ( ncacn_conn ) ;
return ;
}
break ;
default :
DEBUG ( 0 , ( " unknown dcerpc transport: %u! \n " ,
transport ) ) ;
talloc_free ( ncacn_conn ) ;
close ( s ) ;
return ;
}
2011-02-14 10:29:14 +01:00
rc = set_blocking ( s , false ) ;
if ( rc < 0 ) {
2011-02-22 20:40:24 -05:00
DEBUG ( 2 , ( " Failed to set dcerpc socket to non-blocking \n " ) ) ;
2011-02-14 10:29:14 +01:00
talloc_free ( ncacn_conn ) ;
close ( s ) ;
return ;
}
/*
2011-02-22 20:40:24 -05:00
* As soon as we have tstream_bsd_existing_socket set up it will
* take care of closing the socket .
2011-02-14 10:29:14 +01:00
*/
rc = tstream_bsd_existing_socket ( ncacn_conn , s , & ncacn_conn - > tstream ) ;
if ( rc < 0 ) {
2011-02-22 20:40:24 -05:00
DEBUG ( 2 , ( " Failed to create tstream socket for dcerpc \n " ) ) ;
2011-02-14 10:29:14 +01:00
talloc_free ( ncacn_conn ) ;
close ( s ) ;
return ;
}
if ( ncacn_conn - > session_info = = NULL ) {
2018-03-02 14:40:19 +01:00
status = make_session_info_anonymous ( ncacn_conn ,
& ncacn_conn - > session_info ) ;
2011-02-14 10:29:14 +01:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2011-02-22 20:40:24 -05:00
DEBUG ( 2 , ( " Failed to create "
2018-03-02 14:40:19 +01:00
" make_session_info_anonymous - %s \n " ,
2011-02-14 10:29:14 +01:00
nt_errstr ( status ) ) ) ;
talloc_free ( ncacn_conn ) ;
return ;
}
}
rc = make_server_pipes_struct ( ncacn_conn ,
2011-07-21 10:43:56 +02:00
ncacn_conn - > msg_ctx ,
2011-02-14 10:29:14 +01:00
pipe_name ,
2011-03-03 14:20:26 +01:00
ncacn_conn - > transport ,
2017-03-10 12:13:24 +13:00
ncacn_conn - > remote_client_addr ,
ncacn_conn - > local_server_addr ,
2011-02-14 10:29:14 +01:00
ncacn_conn - > session_info ,
& ncacn_conn - > p ,
& sys_errno ) ;
if ( rc < 0 ) {
2011-02-22 20:40:24 -05:00
DEBUG ( 2 , ( " Failed to create pipe struct - %s " ,
strerror ( sys_errno ) ) ) ;
2011-02-14 10:29:14 +01:00
talloc_free ( ncacn_conn ) ;
return ;
}
ncacn_conn - > send_queue = tevent_queue_create ( ncacn_conn ,
2011-02-22 20:40:24 -05:00
" dcerpc send queue " ) ;
2011-02-14 10:29:14 +01:00
if ( ncacn_conn - > send_queue = = NULL ) {
2017-02-24 13:29:12 +13:00
DEBUG ( 0 , ( " Out of memory building dcerpc send queue! \n " ) ) ;
2011-02-14 10:29:14 +01:00
talloc_free ( ncacn_conn ) ;
return ;
}
2011-02-25 13:50:37 +01:00
subreq = dcerpc_read_ncacn_packet_send ( ncacn_conn ,
2011-02-14 10:29:14 +01:00
ncacn_conn - > ev_ctx ,
ncacn_conn - > tstream ) ;
if ( subreq = = NULL ) {
2011-02-22 20:40:24 -05:00
DEBUG ( 2 , ( " Failed to send ncacn packet \n " ) ) ;
2011-02-14 10:29:14 +01:00
talloc_free ( ncacn_conn ) ;
return ;
}
tevent_req_set_callback ( subreq , dcerpc_ncacn_packet_process , ncacn_conn ) ;
2011-02-22 20:40:24 -05:00
DEBUG ( 10 , ( " dcerpc_ncacn_accept done \n " ) ) ;
2011-02-14 10:29:14 +01:00
return ;
}
static void dcerpc_ncacn_packet_process ( struct tevent_req * subreq )
{
struct dcerpc_ncacn_conn * ncacn_conn =
tevent_req_callback_data ( subreq , struct dcerpc_ncacn_conn ) ;
struct _output_data * out = & ncacn_conn - > p - > out_data ;
DATA_BLOB recv_buffer = data_blob_null ;
2011-02-25 13:50:37 +01:00
struct ncacn_packet * pkt ;
2011-02-14 10:29:14 +01:00
uint32_t to_send ;
NTSTATUS status ;
bool ok ;
2011-02-25 13:50:37 +01:00
status = dcerpc_read_ncacn_packet_recv ( subreq , ncacn_conn , & pkt , & recv_buffer ) ;
2011-02-14 10:29:14 +01:00
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2011-03-14 12:29:49 +01:00
if ( ncacn_conn - > disconnect_fn ! = NULL ) {
ok = ncacn_conn - > disconnect_fn ( ncacn_conn - > p ) ;
if ( ! ok ) {
DEBUG ( 3 , ( " Failed to call disconnect function \n " ) ) ;
}
}
2011-02-14 10:29:14 +01:00
goto fail ;
}
2013-12-10 13:59:06 +01:00
/* dcerpc_read_ncacn_packet_recv() returns a full PDU */
ncacn_conn - > p - > in_data . pdu_needed_len = 0 ;
ncacn_conn - > p - > in_data . pdu = recv_buffer ;
if ( dcerpc_get_endian_flag ( & recv_buffer ) & DCERPC_DREP_LE ) {
ncacn_conn - > p - > endian = RPC_LITTLE_ENDIAN ;
} else {
ncacn_conn - > p - > endian = RPC_BIG_ENDIAN ;
2011-02-14 10:29:14 +01:00
}
2013-12-10 13:59:06 +01:00
DEBUG ( 10 , ( " PDU is in %s Endian format! \n " ,
ncacn_conn - > p - > endian ? " Big " : " Little " ) ) ;
process_complete_pdu ( ncacn_conn - > p , pkt ) ;
2011-02-14 10:29:14 +01:00
2013-12-10 13:59:06 +01:00
/* reset pipe state and free PDU */
ncacn_conn - > p - > in_data . pdu . length = 0 ;
2011-02-14 10:29:14 +01:00
talloc_free ( recv_buffer . data ) ;
2011-02-25 13:50:37 +01:00
talloc_free ( pkt ) ;
2011-02-14 10:29:14 +01:00
/*
* This is needed because of the way DCERPC binds work in the RPC
* marshalling code
*/
to_send = out - > frag . length - out - > current_pdu_sent ;
if ( to_send > 0 ) {
DEBUG ( 10 , ( " Current_pdu_len = %u, "
" current_pdu_sent = %u "
" Returning %u bytes \n " ,
( unsigned int ) out - > frag . length ,
( unsigned int ) out - > current_pdu_sent ,
( unsigned int ) to_send ) ) ;
ncacn_conn - > iov = talloc_zero ( ncacn_conn , struct iovec ) ;
if ( ncacn_conn - > iov = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
2011-03-02 10:56:46 +01:00
DEBUG ( 3 , ( " Out of memory! \n " ) ) ;
2011-02-14 10:29:14 +01:00
goto fail ;
}
ncacn_conn - > count = 1 ;
ncacn_conn - > iov [ 0 ] . iov_base = out - > frag . data
+ out - > current_pdu_sent ;
ncacn_conn - > iov [ 0 ] . iov_len = to_send ;
out - > current_pdu_sent + = to_send ;
}
/*
* This condition is false for bind packets , or when we haven ' t yet got
* a full request , and need to wait for more data from the client
*/
while ( out - > data_sent_length < out - > rdata . length ) {
ok = create_next_pdu ( ncacn_conn - > p ) ;
if ( ! ok ) {
DEBUG ( 3 , ( " Failed to create next PDU! \n " ) ) ;
status = NT_STATUS_UNEXPECTED_IO_ERROR ;
goto fail ;
}
ncacn_conn - > iov = talloc_realloc ( ncacn_conn ,
ncacn_conn - > iov ,
struct iovec ,
ncacn_conn - > count + 1 ) ;
if ( ncacn_conn - > iov = = NULL ) {
2011-03-02 10:56:46 +01:00
DEBUG ( 3 , ( " Out of memory! \n " ) ) ;
2011-02-14 10:29:14 +01:00
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
ncacn_conn - > iov [ ncacn_conn - > count ] . iov_base = out - > frag . data ;
ncacn_conn - > iov [ ncacn_conn - > count ] . iov_len = out - > frag . length ;
DEBUG ( 10 , ( " PDU number: %d, PDU Length: %u \n " ,
( unsigned int ) ncacn_conn - > count ,
( unsigned int ) ncacn_conn - > iov [ ncacn_conn - > count ] . iov_len ) ) ;
dump_data ( 11 , ( const uint8_t * ) ncacn_conn - > iov [ ncacn_conn - > count ] . iov_base ,
ncacn_conn - > iov [ ncacn_conn - > count ] . iov_len ) ;
ncacn_conn - > count + + ;
}
/*
* We still don ' t have a complete request , go back and wait for more
* data .
*/
if ( ncacn_conn - > count = = 0 ) {
/* Wait for the next packet */
2011-02-25 13:50:37 +01:00
subreq = dcerpc_read_ncacn_packet_send ( ncacn_conn ,
2011-02-14 10:29:14 +01:00
ncacn_conn - > ev_ctx ,
ncacn_conn - > tstream ) ;
if ( subreq = = NULL ) {
2018-05-04 22:23:01 +02:00
DEBUG ( 2 , ( " Failed to start receiving packets \n " ) ) ;
2011-02-14 10:29:14 +01:00
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
tevent_req_set_callback ( subreq , dcerpc_ncacn_packet_process , ncacn_conn ) ;
return ;
}
DEBUG ( 10 , ( " Sending a total of %u bytes \n " ,
( unsigned int ) ncacn_conn - > p - > out_data . data_sent_length ) ) ;
subreq = tstream_writev_queue_send ( ncacn_conn ,
ncacn_conn - > ev_ctx ,
ncacn_conn - > tstream ,
ncacn_conn - > send_queue ,
ncacn_conn - > iov ,
ncacn_conn - > count ) ;
if ( subreq = = NULL ) {
DEBUG ( 2 , ( " Failed to send packet \n " ) ) ;
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
tevent_req_set_callback ( subreq , dcerpc_ncacn_packet_done , ncacn_conn ) ;
return ;
fail :
2011-03-08 16:10:07 +01:00
DEBUG ( 3 , ( " Terminating client(%s) connection! - '%s' \n " ,
2017-03-10 12:13:24 +13:00
ncacn_conn - > remote_client_name , nt_errstr ( status ) ) ) ;
2011-02-14 10:29:14 +01:00
/* Terminate client connection */
talloc_free ( ncacn_conn ) ;
return ;
}
static void dcerpc_ncacn_packet_done ( struct tevent_req * subreq )
{
struct dcerpc_ncacn_conn * ncacn_conn =
tevent_req_callback_data ( subreq , struct dcerpc_ncacn_conn ) ;
2011-03-08 16:10:07 +01:00
NTSTATUS status = NT_STATUS_OK ;
2011-02-14 10:29:14 +01:00
int sys_errno ;
int rc ;
rc = tstream_writev_queue_recv ( subreq , & sys_errno ) ;
TALLOC_FREE ( subreq ) ;
if ( rc < 0 ) {
DEBUG ( 2 , ( " Writev failed! \n " ) ) ;
2011-03-08 16:10:07 +01:00
status = map_nt_error_from_unix ( sys_errno ) ;
2011-02-14 10:29:14 +01:00
goto fail ;
}
2015-12-23 12:40:58 +01:00
if ( ncacn_conn - > p - > fault_state ! = 0 ) {
DEBUG ( 2 , ( " Disconnect after fault \n " ) ) ;
sys_errno = EINVAL ;
goto fail ;
}
2011-02-14 10:29:14 +01:00
/* clear out any data that may have been left around */
ncacn_conn - > count = 0 ;
TALLOC_FREE ( ncacn_conn - > iov ) ;
data_blob_free ( & ncacn_conn - > p - > in_data . data ) ;
data_blob_free ( & ncacn_conn - > p - > out_data . frag ) ;
data_blob_free ( & ncacn_conn - > p - > out_data . rdata ) ;
2011-07-26 12:07:20 +02:00
talloc_free_children ( ncacn_conn - > p - > mem_ctx ) ;
2011-02-14 10:29:14 +01:00
/* Wait for the next packet */
2011-02-25 13:50:37 +01:00
subreq = dcerpc_read_ncacn_packet_send ( ncacn_conn ,
2011-02-14 10:29:14 +01:00
ncacn_conn - > ev_ctx ,
ncacn_conn - > tstream ) ;
if ( subreq = = NULL ) {
2018-05-04 22:23:01 +02:00
DEBUG ( 2 , ( " Failed to start receiving packets \n " ) ) ;
2011-03-08 16:10:07 +01:00
status = NT_STATUS_NO_MEMORY ;
2011-02-14 10:29:14 +01:00
goto fail ;
}
tevent_req_set_callback ( subreq , dcerpc_ncacn_packet_process , ncacn_conn ) ;
return ;
fail :
2011-03-08 16:10:07 +01:00
DEBUG ( 3 , ( " Terminating client(%s) connection! - '%s' \n " ,
2017-03-10 12:13:24 +13:00
ncacn_conn - > remote_client_name , nt_errstr ( status ) ) ) ;
2011-02-14 10:29:14 +01:00
/* Terminate client connection */
talloc_free ( ncacn_conn ) ;
return ;
}
/* vim: set ts=8 sw=8 noet cindent syntax=c.doxygen: */