2009-01-22 20:34:06 +03:00
/*
* Unix SMB / CIFS implementation .
* RPC client transport over named pipes
* Copyright ( C ) Volker Lendecke 2009
*
* 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-04-28 19:38:09 +04:00
# include "../lib/util/tevent_ntstatus.h"
2011-03-31 02:34:28 +04:00
# include "rpc_client/rpc_transport.h"
2013-09-18 12:59:14 +04:00
# include "librpc/ndr/ndr_table.h"
2013-09-13 14:55:34 +04:00
# include "libcli/smb/smbXcli_base.h"
2013-09-13 17:50:10 +04:00
# include "libcli/smb/tstream_smbXcli_np.h"
2013-09-13 14:55:34 +04:00
# include "client.h"
2009-01-22 20:34:06 +03:00
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_RPC_CLI
2009-01-30 00:41:33 +03:00
struct rpc_transport_np_init_state {
struct rpc_cli_transport * transport ;
2014-03-03 22:49:35 +04:00
int retries ;
struct tevent_context * ev ;
struct smbXcli_conn * conn ;
int timeout ;
struct timeval abs_timeout ;
const char * pipe_name ;
struct smbXcli_session * session ;
struct smbXcli_tcon * tcon ;
uint16_t pid ;
2009-01-30 00:41:33 +03:00
} ;
2009-03-29 22:00:58 +04:00
static void rpc_transport_np_init_pipe_open ( struct tevent_req * subreq ) ;
2009-01-30 00:41:33 +03:00
2009-04-08 23:33:01 +04:00
struct tevent_req * rpc_transport_np_init_send ( TALLOC_CTX * mem_ctx ,
2013-02-18 12:57:22 +04:00
struct tevent_context * ev ,
2009-04-08 23:33:01 +04:00
struct cli_state * cli ,
2013-05-24 15:56:53 +04:00
const struct ndr_interface_table * table )
2009-01-22 20:34:06 +03:00
{
2010-09-06 19:31:15 +04:00
struct tevent_req * req ;
2009-01-30 00:41:33 +03:00
struct rpc_transport_np_init_state * state ;
2010-09-06 19:31:15 +04:00
struct tevent_req * subreq ;
2009-01-22 20:34:06 +03:00
2009-04-08 23:33:01 +04:00
req = tevent_req_create ( mem_ctx , & state ,
struct rpc_transport_np_init_state ) ;
if ( req = = NULL ) {
2009-01-30 00:41:33 +03:00
return NULL ;
2009-01-22 20:34:06 +03:00
}
2009-01-30 00:41:33 +03:00
2013-09-13 14:55:34 +04:00
if ( smbXcli_conn_protocol ( cli - > conn ) > = PROTOCOL_SMB2_02 ) {
2014-03-03 22:49:35 +04:00
state - > tcon = cli - > smb2 . tcon ;
state - > session = cli - > smb2 . session ;
2013-09-13 14:55:34 +04:00
} else {
2014-03-03 22:49:35 +04:00
state - > tcon = cli - > smb1 . tcon ;
state - > session = cli - > smb1 . session ;
state - > pid = cli - > smb1 . pid ;
2013-09-13 14:55:34 +04:00
}
2014-03-03 22:49:35 +04:00
state - > ev = ev ;
state - > conn = cli - > conn ;
state - > timeout = cli - > timeout ;
state - > abs_timeout = timeval_current_ofs_msec ( cli - > timeout ) ;
state - > pipe_name = dcerpc_default_transport_endpoint ( state , NCACN_NP ,
table ) ;
if ( tevent_req_nomem ( state - > pipe_name , req ) ) {
2009-04-08 23:33:01 +04:00
return tevent_req_post ( req , ev ) ;
2009-01-22 20:34:06 +03:00
}
2014-03-03 22:49:35 +04:00
while ( state - > pipe_name [ 0 ] = = ' \\ ' ) {
state - > pipe_name + + ;
2012-05-28 20:43:29 +04:00
}
2014-03-03 22:49:35 +04:00
subreq = tstream_smbXcli_np_open_send ( state , ev , state - > conn ,
state - > session , state - > tcon ,
state - > pid , state - > timeout ,
state - > pipe_name ) ;
2009-04-08 23:33:01 +04:00
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
2009-01-30 00:41:33 +03:00
}
2010-09-06 19:31:15 +04:00
tevent_req_set_callback ( subreq , rpc_transport_np_init_pipe_open , req ) ;
2009-04-08 23:33:01 +04:00
return req ;
2009-01-30 00:41:33 +03:00
}
2014-03-03 22:49:35 +04:00
static void rpc_transport_np_init_pipe_open_retry ( struct tevent_context * ev ,
struct tevent_timer * te ,
struct timeval t ,
void * priv_data )
{
struct tevent_req * subreq ;
struct tevent_req * req = talloc_get_type ( priv_data , struct tevent_req ) ;
struct rpc_transport_np_init_state * state = tevent_req_data (
req , struct rpc_transport_np_init_state ) ;
subreq = tstream_smbXcli_np_open_send ( state , ev ,
state - > conn ,
state - > session ,
state - > tcon ,
state - > pid ,
state - > timeout ,
state - > pipe_name ) ;
if ( tevent_req_nomem ( subreq , req ) ) {
return ;
}
tevent_req_set_callback ( subreq , rpc_transport_np_init_pipe_open , req ) ;
state - > retries + + ;
}
2009-03-29 22:00:58 +04:00
static void rpc_transport_np_init_pipe_open ( struct tevent_req * subreq )
2009-01-30 00:41:33 +03:00
{
2009-04-08 23:33:01 +04:00
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct rpc_transport_np_init_state * state = tevent_req_data (
req , struct rpc_transport_np_init_state ) ;
2009-01-30 00:41:33 +03:00
NTSTATUS status ;
2010-09-06 19:31:15 +04:00
struct tstream_context * stream ;
2009-01-30 00:41:33 +03:00
2013-11-29 15:24:50 +04:00
status = tstream_smbXcli_np_open_recv ( subreq , state , & stream ) ;
2009-01-30 00:41:33 +03:00
TALLOC_FREE ( subreq ) ;
2014-03-03 22:49:35 +04:00
if ( NT_STATUS_EQUAL ( status , NT_STATUS_PIPE_NOT_AVAILABLE )
& & ( ! timeval_expired ( & state - > abs_timeout ) ) ) {
struct tevent_timer * te ;
/*
* Retry on STATUS_PIPE_NOT_AVAILABLE , Windows starts some
* servers ( FssagentRpc ) on demand .
*/
DEBUG ( 2 , ( " RPC pipe %s not available, retry %d \n " ,
state - > pipe_name , state - > retries ) ) ;
te = tevent_add_timer ( state - > ev , state ,
timeval_current_ofs_msec ( 100 * state - > retries ) ,
rpc_transport_np_init_pipe_open_retry , req ) ;
if ( tevent_req_nomem ( te , req ) ) {
2015-08-03 11:31:58 +03:00
DEBUG ( 2 , ( " Failed to create asynchronous "
" tevent_timer " ) ) ;
2014-03-03 22:49:35 +04:00
}
return ;
} else if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-04-08 23:33:01 +04:00
tevent_req_nterror ( req , status ) ;
2009-01-30 00:41:33 +03:00
return ;
}
2010-09-06 19:31:15 +04:00
status = rpc_transport_tstream_init ( state ,
& stream ,
& state - > transport ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
tevent_req_nterror ( req , status ) ;
return ;
}
2009-04-08 23:33:01 +04:00
tevent_req_done ( req ) ;
2009-01-30 00:41:33 +03:00
}
2009-04-08 23:33:01 +04:00
NTSTATUS rpc_transport_np_init_recv ( struct tevent_req * req ,
2009-01-30 00:41:33 +03:00
TALLOC_CTX * mem_ctx ,
struct rpc_cli_transport * * presult )
{
2009-04-08 23:33:01 +04:00
struct rpc_transport_np_init_state * state = tevent_req_data (
req , struct rpc_transport_np_init_state ) ;
2009-01-30 00:41:33 +03:00
NTSTATUS status ;
2009-01-22 20:34:06 +03:00
2009-04-08 23:33:01 +04:00
if ( tevent_req_is_nterror ( req , & status ) ) {
2009-01-30 00:41:33 +03:00
return status ;
2009-01-22 20:34:06 +03:00
}
2009-01-30 00:41:33 +03:00
* presult = talloc_move ( mem_ctx , & state - > transport ) ;
2009-01-22 20:34:06 +03:00
return NT_STATUS_OK ;
}
2009-01-30 00:41:33 +03:00
NTSTATUS rpc_transport_np_init ( TALLOC_CTX * mem_ctx , struct cli_state * cli ,
2013-05-24 15:52:05 +04:00
const struct ndr_interface_table * table ,
2009-01-30 00:41:33 +03:00
struct rpc_cli_transport * * presult )
{
TALLOC_CTX * frame = talloc_stackframe ( ) ;
2013-02-18 12:57:22 +04:00
struct tevent_context * ev ;
2009-04-08 23:33:01 +04:00
struct tevent_req * req ;
NTSTATUS status = NT_STATUS_OK ;
2009-01-30 00:41:33 +03:00
2013-02-18 12:17:45 +04:00
ev = samba_tevent_context_init ( frame ) ;
2009-01-30 00:41:33 +03:00
if ( ev = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
2013-05-24 15:56:53 +04:00
req = rpc_transport_np_init_send ( frame , ev , cli , table ) ;
2009-01-30 00:41:33 +03:00
if ( req = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
2015-04-28 17:00:06 +03:00
if ( ! tevent_req_poll_ntstatus ( req , ev , & status ) ) {
2009-04-08 23:33:01 +04:00
goto fail ;
2009-01-30 00:41:33 +03:00
}
status = rpc_transport_np_init_recv ( req , mem_ctx , presult ) ;
fail :
TALLOC_FREE ( frame ) ;
return status ;
}