2009-01-22 18:34:06 +01: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 17:38:09 +02:00
# include "../lib/util/tevent_ntstatus.h"
2021-04-02 12:20:38 +02:00
# include "librpc/rpc/dcerpc_util.h"
2011-03-31 00:34:28 +02:00
# include "rpc_client/rpc_transport.h"
2013-09-18 10:59:14 +02:00
# include "librpc/ndr/ndr_table.h"
2013-09-13 12:55:34 +02:00
# include "libcli/smb/smbXcli_base.h"
2013-09-13 15:50:10 +02:00
# include "libcli/smb/tstream_smbXcli_np.h"
2013-09-13 12:55:34 +02:00
# include "client.h"
2009-01-22 18:34:06 +01:00
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_RPC_CLI
2009-01-29 22:41:33 +01:00
struct rpc_transport_np_init_state {
struct rpc_cli_transport * transport ;
2014-03-03 19:49:35 +01: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-29 22:41:33 +01:00
} ;
2009-03-29 20:00:58 +02:00
static void rpc_transport_np_init_pipe_open ( struct tevent_req * subreq ) ;
2009-01-29 22:41:33 +01:00
2009-04-08 21:33:01 +02:00
struct tevent_req * rpc_transport_np_init_send ( TALLOC_CTX * mem_ctx ,
2013-02-18 09:57:22 +01:00
struct tevent_context * ev ,
2009-04-08 21:33:01 +02:00
struct cli_state * cli ,
2024-09-20 00:24:05 +02:00
const char * pipe_name )
2009-01-22 18:34:06 +01:00
{
2010-09-06 17:31:15 +02:00
struct tevent_req * req ;
2009-01-29 22:41:33 +01:00
struct rpc_transport_np_init_state * state ;
2010-09-06 17:31:15 +02:00
struct tevent_req * subreq ;
2009-01-22 18:34:06 +01:00
2009-04-08 21:33:01 +02:00
req = tevent_req_create ( mem_ctx , & state ,
struct rpc_transport_np_init_state ) ;
if ( req = = NULL ) {
2009-01-29 22:41:33 +01:00
return NULL ;
2009-01-22 18:34:06 +01:00
}
2009-01-29 22:41:33 +01:00
2013-09-13 12:55:34 +02:00
if ( smbXcli_conn_protocol ( cli - > conn ) > = PROTOCOL_SMB2_02 ) {
2014-03-03 19:49:35 +01:00
state - > tcon = cli - > smb2 . tcon ;
state - > session = cli - > smb2 . session ;
2013-09-13 12:55:34 +02:00
} else {
2014-03-03 19:49:35 +01:00
state - > tcon = cli - > smb1 . tcon ;
state - > session = cli - > smb1 . session ;
state - > pid = cli - > smb1 . pid ;
2013-09-13 12:55:34 +02:00
}
2014-03-03 19:49:35 +01:00
state - > ev = ev ;
state - > conn = cli - > conn ;
state - > timeout = cli - > timeout ;
state - > abs_timeout = timeval_current_ofs_msec ( cli - > timeout ) ;
2024-09-20 00:24:05 +02:00
state - > pipe_name = talloc_strdup ( state , pipe_name ) ;
2014-03-03 19:49:35 +01:00
if ( tevent_req_nomem ( state - > pipe_name , req ) ) {
2009-04-08 21:33:01 +02:00
return tevent_req_post ( req , ev ) ;
2009-01-22 18:34:06 +01:00
}
2014-03-03 19:49:35 +01:00
while ( state - > pipe_name [ 0 ] = = ' \\ ' ) {
state - > pipe_name + + ;
2012-05-28 18:43:29 +02:00
}
2014-03-03 19:49:35 +01: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 21:33:01 +02:00
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
2009-01-29 22:41:33 +01:00
}
2010-09-06 17:31:15 +02:00
tevent_req_set_callback ( subreq , rpc_transport_np_init_pipe_open , req ) ;
2009-04-08 21:33:01 +02:00
return req ;
2009-01-29 22:41:33 +01:00
}
2014-03-03 19:49:35 +01: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 20:00:58 +02:00
static void rpc_transport_np_init_pipe_open ( struct tevent_req * subreq )
2009-01-29 22:41:33 +01:00
{
2009-04-08 21:33:01 +02: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-29 22:41:33 +01:00
NTSTATUS status ;
2010-09-06 17:31:15 +02:00
struct tstream_context * stream ;
2009-01-29 22:41:33 +01:00
2013-11-29 12:24:50 +01:00
status = tstream_smbXcli_np_open_recv ( subreq , state , & stream ) ;
2009-01-29 22:41:33 +01:00
TALLOC_FREE ( subreq ) ;
2014-03-03 19:49:35 +01: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 14:01:58 +05:30
DEBUG ( 2 , ( " Failed to create asynchronous "
2023-08-07 16:47:47 +12:00
" tevent_timer \n " ) ) ;
2014-03-03 19:49:35 +01:00
}
return ;
2021-08-17 16:52:24 +02:00
}
if ( tevent_req_nterror ( req , status ) ) {
2009-01-29 22:41:33 +01:00
return ;
}
2010-09-06 17:31:15 +02:00
status = rpc_transport_tstream_init ( state ,
& stream ,
& state - > transport ) ;
2021-08-17 16:52:24 +02:00
if ( tevent_req_nterror ( req , status ) ) {
2010-09-06 17:31:15 +02:00
return ;
}
2009-04-08 21:33:01 +02:00
tevent_req_done ( req ) ;
2009-01-29 22:41:33 +01:00
}
2009-04-08 21:33:01 +02:00
NTSTATUS rpc_transport_np_init_recv ( struct tevent_req * req ,
2009-01-29 22:41:33 +01:00
TALLOC_CTX * mem_ctx ,
struct rpc_cli_transport * * presult )
{
2009-04-08 21:33:01 +02:00
struct rpc_transport_np_init_state * state = tevent_req_data (
req , struct rpc_transport_np_init_state ) ;
2009-01-29 22:41:33 +01:00
NTSTATUS status ;
2009-01-22 18:34:06 +01:00
2009-04-08 21:33:01 +02:00
if ( tevent_req_is_nterror ( req , & status ) ) {
2009-01-29 22:41:33 +01:00
return status ;
2009-01-22 18:34:06 +01:00
}
2009-01-29 22:41:33 +01:00
* presult = talloc_move ( mem_ctx , & state - > transport ) ;
2009-01-22 18:34:06 +01:00
return NT_STATUS_OK ;
}