2009-01-29 22:54:55 +01:00
/*
* Unix SMB / CIFS implementation .
* RPC client transport over named pipes to a child smbd
* 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"
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_RPC_CLI
/**
* struct rpc_cli_smbd_conn represents a forked smbd . This structure should
* exist only once per process which does the rpc calls .
*
* RPC pipe handles can be attached to this smbd connection with
* rpc_pipe_open_local ( ) .
*
* For this to work right , we can not use rpc_transport_np directly , because
* the child smbd wants to write its DEBUG output somewhere . We redirect the
* child ' s output to rpc_cli_smbd_conn - > stdout_fd . While the RPC calls are
* active , we have an event context available and attach a fd event to the
* stdout_df .
*/
struct rpc_cli_smbd_conn {
/**
* The smb connection to handle the named pipe traffic over
*/
struct cli_state * cli ;
/**
* Attached to stdout in the forked smbd , this is where smbd will
* print its DEBUG .
*/
int stdout_fd ;
/**
* Custom callback provided by the owner of the
* rpc_cli_smbd_conn . Here we send the smbd DEBUG output . Can be NULL .
*/
struct {
void ( * fn ) ( char * buf , size_t len , void * priv ) ;
void * priv ;
} stdout_callback ;
} ;
/**
* Event handler to be called whenever the forked smbd prints debugging
* output .
*/
static void rpc_cli_smbd_stdout_reader ( struct event_context * ev ,
struct fd_event * fde ,
uint16_t flags , void * priv )
{
struct rpc_cli_smbd_conn * conn = talloc_get_type_abort (
priv , struct rpc_cli_smbd_conn ) ;
char buf [ 1024 ] ;
ssize_t nread ;
if ( ( flags & EVENT_FD_READ ) = = 0 ) {
return ;
}
nread = read ( conn - > stdout_fd , buf , sizeof ( buf ) - 1 ) ;
if ( nread < 0 ) {
DEBUG ( 0 , ( " Could not read from smbd stdout: %s \n " ,
strerror ( errno ) ) ) ;
TALLOC_FREE ( fde ) ;
return ;
}
if ( nread = = 0 ) {
DEBUG ( 0 , ( " EOF from smbd stdout \n " ) ) ;
TALLOC_FREE ( fde ) ;
return ;
}
2009-02-12 12:11:16 +01:00
buf [ nread ] = ' \0 ' ;
2009-01-29 22:54:55 +01:00
if ( conn - > stdout_callback . fn ! = NULL ) {
conn - > stdout_callback . fn ( buf , nread ,
conn - > stdout_callback . priv ) ;
}
}
/**
* struct rpc_transport_smbd_state is the link from a struct rpc_pipe_client
* to the rpc_cli_smbd_conn . We use a named pipe transport as a subtransport .
*/
struct rpc_transport_smbd_state {
struct rpc_cli_smbd_conn * conn ;
struct rpc_cli_transport * sub_transp ;
} ;
static int rpc_cli_smbd_conn_destructor ( struct rpc_cli_smbd_conn * conn )
{
if ( conn - > cli ! = NULL ) {
cli_shutdown ( conn - > cli ) ;
conn - > cli = NULL ;
}
if ( conn - > stdout_fd ! = - 1 ) {
close ( conn - > stdout_fd ) ;
conn - > stdout_fd = - 1 ;
}
return 0 ;
}
/*
* Do the negprot / sesssetup / tcon to an anonymous ipc $ connection
*/
struct get_anon_ipc_state {
struct event_context * ev ;
struct cli_state * cli ;
} ;
2009-03-29 18:59:56 +02:00
static void get_anon_ipc_negprot_done ( struct tevent_req * subreq ) ;
2009-03-29 17:42:13 +02:00
static void get_anon_ipc_sesssetup_done ( struct tevent_req * subreq ) ;
2009-03-29 19:47:18 +02:00
static void get_anon_ipc_tcon_done ( struct tevent_req * subreq ) ;
2009-01-29 22:54:55 +01:00
2009-04-08 21:42:01 +02:00
static struct tevent_req * get_anon_ipc_send ( TALLOC_CTX * mem_ctx ,
struct event_context * ev ,
struct cli_state * cli )
2009-01-29 22:54:55 +01:00
{
2009-04-08 21:42:01 +02:00
struct tevent_req * req , * subreq ;
2009-01-29 22:54:55 +01:00
struct get_anon_ipc_state * state ;
2009-04-08 21:42:01 +02:00
req = tevent_req_create ( mem_ctx , & state , struct get_anon_ipc_state ) ;
if ( req = = NULL ) {
2009-01-29 22:54:55 +01:00
return NULL ;
}
state - > ev = ev ;
state - > cli = cli ;
subreq = cli_negprot_send ( state , ev , cli ) ;
2009-04-08 21:42:01 +02:00
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
2009-01-29 22:54:55 +01:00
}
2009-04-08 21:42:01 +02:00
tevent_req_set_callback ( subreq , get_anon_ipc_negprot_done , req ) ;
return req ;
2009-01-29 22:54:55 +01:00
}
2009-03-29 18:59:56 +02:00
static void get_anon_ipc_negprot_done ( struct tevent_req * subreq )
2009-01-29 22:54:55 +01:00
{
2009-04-08 21:42:01 +02:00
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct get_anon_ipc_state * state = tevent_req_data (
req , struct get_anon_ipc_state ) ;
2009-01-29 22:54:55 +01:00
NTSTATUS status ;
status = cli_negprot_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-04-08 21:42:01 +02:00
tevent_req_nterror ( req , status ) ;
2009-01-29 22:54:55 +01:00
return ;
}
2009-03-29 19:47:18 +02:00
subreq = cli_session_setup_guest_send ( state , state - > ev , state - > cli ) ;
2009-04-08 21:42:01 +02:00
if ( tevent_req_nomem ( subreq , req ) ) {
2009-01-29 22:54:55 +01:00
return ;
}
2009-03-29 19:47:18 +02:00
tevent_req_set_callback ( subreq , get_anon_ipc_sesssetup_done , req ) ;
2009-01-29 22:54:55 +01:00
}
2009-03-29 17:42:13 +02:00
static void get_anon_ipc_sesssetup_done ( struct tevent_req * subreq )
2009-01-29 22:54:55 +01:00
{
2009-04-08 21:42:01 +02:00
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct get_anon_ipc_state * state = tevent_req_data (
req , struct get_anon_ipc_state ) ;
2009-01-29 22:54:55 +01:00
NTSTATUS status ;
status = cli_session_setup_guest_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-04-08 21:42:01 +02:00
tevent_req_nterror ( req , status ) ;
2009-01-29 22:54:55 +01:00
return ;
}
2009-03-29 19:47:18 +02:00
subreq = cli_tcon_andx_send ( state , state - > ev , state - > cli ,
2009-01-29 22:54:55 +01:00
" IPC$ " , " IPC " , NULL , 0 ) ;
2009-04-08 21:42:01 +02:00
if ( tevent_req_nomem ( subreq , req ) ) {
2009-01-29 22:54:55 +01:00
return ;
}
2009-03-29 19:47:18 +02:00
tevent_req_set_callback ( subreq , get_anon_ipc_tcon_done , req ) ;
2009-01-29 22:54:55 +01:00
}
2009-03-29 19:47:18 +02:00
static void get_anon_ipc_tcon_done ( struct tevent_req * subreq )
2009-01-29 22:54:55 +01:00
{
2009-04-08 21:42:01 +02:00
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
2009-01-29 22:54:55 +01:00
NTSTATUS status ;
status = cli_tcon_andx_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-04-08 21:42:01 +02:00
tevent_req_nterror ( req , status ) ;
2009-01-29 22:54:55 +01:00
return ;
}
2009-04-08 21:42:01 +02:00
tevent_req_done ( req ) ;
2009-01-29 22:54:55 +01:00
}
2009-04-08 21:42:01 +02:00
static NTSTATUS get_anon_ipc_recv ( struct tevent_req * req )
2009-01-29 22:54:55 +01:00
{
2009-04-08 21:42:01 +02:00
return tevent_req_simple_recv_ntstatus ( req ) ;
2009-01-29 22:54:55 +01:00
}
struct rpc_cli_smbd_conn_init_state {
struct event_context * ev ;
struct rpc_cli_smbd_conn * conn ;
} ;
2009-04-08 21:42:01 +02:00
static void rpc_cli_smbd_conn_init_done ( struct tevent_req * subreq ) ;
2009-01-29 22:54:55 +01:00
2009-04-08 21:53:16 +02:00
struct tevent_req * rpc_cli_smbd_conn_init_send ( TALLOC_CTX * mem_ctx ,
struct event_context * ev ,
void ( * stdout_callback ) ( char * buf ,
size_t len ,
void * priv ) ,
void * priv )
2009-01-29 22:54:55 +01:00
{
2009-04-08 21:53:16 +02:00
struct tevent_req * req , * subreq ;
2009-01-29 22:54:55 +01:00
struct rpc_cli_smbd_conn_init_state * state ;
int smb_sock [ 2 ] ;
int stdout_pipe [ 2 ] ;
NTSTATUS status ;
pid_t pid ;
int ret ;
smb_sock [ 0 ] = smb_sock [ 1 ] = stdout_pipe [ 0 ] = stdout_pipe [ 1 ] = - 1 ;
2009-04-08 21:53:16 +02:00
req = tevent_req_create ( mem_ctx , & state ,
struct rpc_cli_smbd_conn_init_state ) ;
if ( req = = NULL ) {
2009-01-29 22:54:55 +01:00
return NULL ;
}
state - > ev = ev ;
state - > conn = talloc ( state , struct rpc_cli_smbd_conn ) ;
2009-04-08 21:53:16 +02:00
if ( tevent_req_nomem ( state - > conn , req ) ) {
return tevent_req_post ( req , ev ) ;
2009-01-29 22:54:55 +01:00
}
state - > conn - > cli = cli_initialise ( ) ;
2009-04-08 21:53:16 +02:00
if ( tevent_req_nomem ( state - > conn - > cli , req ) ) {
return tevent_req_post ( req , ev ) ;
2009-01-29 22:54:55 +01:00
}
state - > conn - > stdout_fd = - 1 ;
state - > conn - > stdout_callback . fn = stdout_callback ;
state - > conn - > stdout_callback . priv = priv ;
talloc_set_destructor ( state - > conn , rpc_cli_smbd_conn_destructor ) ;
ret = socketpair ( AF_UNIX , SOCK_STREAM , 0 , smb_sock ) ;
if ( ret = = - 1 ) {
status = map_nt_error_from_unix ( errno ) ;
goto post_status ;
}
ret = pipe ( stdout_pipe ) ;
if ( ret = = - 1 ) {
status = map_nt_error_from_unix ( errno ) ;
goto post_status ;
}
pid = sys_fork ( ) ;
if ( pid = = - 1 ) {
status = map_nt_error_from_unix ( errno ) ;
goto post_status ;
}
if ( pid = = 0 ) {
char * smbd_cmd ;
close ( smb_sock [ 0 ] ) ;
close ( stdout_pipe [ 0 ] ) ;
close ( 0 ) ;
if ( dup ( smb_sock [ 1 ] ) = = - 1 ) {
exit ( 1 ) ;
}
close ( smb_sock [ 1 ] ) ;
close ( 1 ) ;
if ( dup ( stdout_pipe [ 1 ] ) = = - 1 ) {
exit ( 1 ) ;
}
close ( stdout_pipe [ 1 ] ) ;
smbd_cmd = getenv ( " SMB_PATH " ) ;
if ( ( smbd_cmd = = NULL )
& & ( asprintf ( & smbd_cmd , " %s/smbd " , get_dyn_SBINDIR ( ) )
= = - 1 ) ) {
printf ( " no memory " ) ;
exit ( 1 ) ;
}
2009-02-12 20:41:49 +01:00
if ( asprintf ( & smbd_cmd , " %s -F -S -d %d " , smbd_cmd ,
DEBUGLEVEL ) = = - 1 ) {
2009-01-29 22:54:55 +01:00
printf ( " no memory " ) ;
exit ( 1 ) ;
}
exit ( system ( smbd_cmd ) ) ;
}
state - > conn - > cli - > fd = smb_sock [ 0 ] ;
smb_sock [ 0 ] = - 1 ;
close ( smb_sock [ 1 ] ) ;
smb_sock [ 1 ] = - 1 ;
state - > conn - > stdout_fd = stdout_pipe [ 0 ] ;
stdout_pipe [ 0 ] = - 1 ;
close ( stdout_pipe [ 1 ] ) ;
stdout_pipe [ 1 ] = - 1 ;
subreq = get_anon_ipc_send ( state , ev , state - > conn - > cli ) ;
2009-04-08 21:53:16 +02:00
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
2009-01-29 22:54:55 +01:00
}
2009-02-12 12:12:04 +01:00
if ( event_add_fd ( ev , state , state - > conn - > stdout_fd , EVENT_FD_READ ,
2009-01-29 22:54:55 +01:00
rpc_cli_smbd_stdout_reader , state - > conn ) = = NULL ) {
2009-04-08 21:53:16 +02:00
status = NT_STATUS_NO_MEMORY ;
goto post_status ;
2009-01-29 22:54:55 +01:00
}
2009-04-08 21:53:16 +02:00
tevent_req_set_callback ( subreq , rpc_cli_smbd_conn_init_done , req ) ;
return req ;
2009-01-29 22:54:55 +01:00
post_status :
if ( smb_sock [ 0 ] ! = - 1 ) {
close ( smb_sock [ 0 ] ) ;
}
if ( smb_sock [ 1 ] ! = - 1 ) {
close ( smb_sock [ 1 ] ) ;
}
if ( stdout_pipe [ 0 ] ! = - 1 ) {
close ( stdout_pipe [ 0 ] ) ;
}
if ( stdout_pipe [ 1 ] ! = - 1 ) {
close ( stdout_pipe [ 1 ] ) ;
}
2009-04-08 21:53:16 +02:00
tevent_req_nterror ( req , status ) ;
return tevent_req_post ( req , ev ) ;
2009-01-29 22:54:55 +01:00
}
2009-04-08 21:42:01 +02:00
static void rpc_cli_smbd_conn_init_done ( struct tevent_req * subreq )
2009-01-29 22:54:55 +01:00
{
2009-04-08 21:53:16 +02:00
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
2009-01-29 22:54:55 +01:00
NTSTATUS status ;
status = get_anon_ipc_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-04-08 21:53:16 +02:00
tevent_req_nterror ( req , status ) ;
2009-01-29 22:54:55 +01:00
return ;
}
2009-04-08 21:53:16 +02:00
tevent_req_done ( req ) ;
2009-01-29 22:54:55 +01:00
}
2009-04-08 21:53:16 +02:00
NTSTATUS rpc_cli_smbd_conn_init_recv ( struct tevent_req * req ,
2009-01-29 22:54:55 +01:00
TALLOC_CTX * mem_ctx ,
struct rpc_cli_smbd_conn * * pconn )
{
2009-04-08 21:53:16 +02:00
struct rpc_cli_smbd_conn_init_state * state = tevent_req_data (
req , struct rpc_cli_smbd_conn_init_state ) ;
2009-01-29 22:54:55 +01:00
NTSTATUS status ;
2009-04-08 21:53:16 +02:00
if ( tevent_req_is_nterror ( req , & status ) ) {
2009-01-29 22:54:55 +01:00
return status ;
}
* pconn = talloc_move ( mem_ctx , & state - > conn ) ;
return NT_STATUS_OK ;
}
NTSTATUS rpc_cli_smbd_conn_init ( TALLOC_CTX * mem_ctx ,
struct rpc_cli_smbd_conn * * pconn ,
void ( * stdout_callback ) ( char * buf ,
size_t len ,
void * priv ) ,
void * priv )
{
TALLOC_CTX * frame = talloc_stackframe ( ) ;
struct event_context * ev ;
2009-04-08 21:53:16 +02:00
struct tevent_req * req ;
2009-01-29 22:54:55 +01:00
NTSTATUS status ;
ev = event_context_init ( frame ) ;
if ( ev = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
req = rpc_cli_smbd_conn_init_send ( frame , ev , stdout_callback , priv ) ;
if ( req = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
2009-04-08 21:53:16 +02:00
if ( ! tevent_req_poll ( req , ev ) ) {
status = map_nt_error_from_unix ( errno ) ;
goto fail ;
2009-01-29 22:54:55 +01:00
}
status = rpc_cli_smbd_conn_init_recv ( req , mem_ctx , pconn ) ;
fail :
TALLOC_FREE ( frame ) ;
return status ;
}
2010-03-25 15:00:38 +01:00
static void rpc_smbd_disconnect ( struct rpc_transport_smbd_state * transp )
{
if ( transp = = NULL ) {
return ;
}
if ( transp - > conn = = NULL ) {
return ;
}
if ( transp - > conn - > cli = = NULL ) {
return ;
}
if ( transp - > conn - > cli - > fd ! = - 1 ) {
close ( transp - > conn - > cli - > fd ) ;
transp - > conn - > cli - > fd = - 1 ;
}
transp - > conn = NULL ;
}
static bool rpc_smbd_is_connected ( void * priv )
{
struct rpc_transport_smbd_state * transp = talloc_get_type_abort (
priv , struct rpc_transport_smbd_state ) ;
bool ok ;
if ( transp - > conn = = NULL ) {
return false ;
}
if ( transp - > sub_transp = = NULL ) {
return false ;
}
ok = transp - > sub_transp - > is_connected ( transp - > sub_transp - > priv ) ;
if ( ! ok ) {
rpc_smbd_disconnect ( transp ) ;
return false ;
}
return true ;
}
2010-03-25 15:51:51 +01:00
static unsigned int rpc_smbd_set_timeout ( void * priv , unsigned int timeout )
{
struct rpc_transport_smbd_state * transp = talloc_get_type_abort (
priv , struct rpc_transport_smbd_state ) ;
bool ok ;
ok = rpc_smbd_is_connected ( transp ) ;
if ( ! ok ) {
return 0 ;
}
if ( transp - > sub_transp - > set_timeout = = NULL ) {
return 0 ;
}
return transp - > sub_transp - > set_timeout ( transp - > sub_transp - > priv , timeout ) ;
}
2009-01-29 22:54:55 +01:00
struct rpc_smbd_write_state {
2010-03-25 15:00:38 +01:00
struct rpc_transport_smbd_state * transp ;
2009-01-29 22:54:55 +01:00
ssize_t written ;
} ;
2009-03-23 23:20:03 +01:00
static void rpc_smbd_write_done ( struct tevent_req * subreq ) ;
2009-01-29 22:54:55 +01:00
2009-03-23 23:20:03 +01:00
static struct tevent_req * rpc_smbd_write_send ( TALLOC_CTX * mem_ctx ,
struct event_context * ev ,
const uint8_t * data , size_t size ,
void * priv )
2009-01-29 22:54:55 +01:00
{
struct rpc_transport_smbd_state * transp = talloc_get_type_abort (
priv , struct rpc_transport_smbd_state ) ;
2009-03-23 23:20:03 +01:00
struct tevent_req * req , * subreq ;
2009-01-29 22:54:55 +01:00
struct rpc_smbd_write_state * state ;
2010-03-25 15:00:38 +01:00
bool ok ;
2009-01-29 22:54:55 +01:00
2009-03-23 23:20:03 +01:00
req = tevent_req_create ( mem_ctx , & state , struct rpc_smbd_write_state ) ;
if ( req = = NULL ) {
2009-01-29 22:54:55 +01:00
return NULL ;
}
2010-03-25 15:00:38 +01:00
ok = rpc_smbd_is_connected ( transp ) ;
if ( ! ok ) {
tevent_req_nterror ( req , NT_STATUS_CONNECTION_INVALID ) ;
return tevent_req_post ( req , ev ) ;
}
state - > transp = transp ;
2009-01-29 22:54:55 +01:00
subreq = transp - > sub_transp - > write_send ( state , ev , data , size ,
transp - > sub_transp - > priv ) ;
if ( subreq = = NULL ) {
goto fail ;
}
2009-02-12 12:12:04 +01:00
if ( event_add_fd ( ev , state , transp - > conn - > stdout_fd , EVENT_FD_READ ,
2009-01-29 22:54:55 +01:00
rpc_cli_smbd_stdout_reader , transp - > conn ) = = NULL ) {
goto fail ;
}
2009-03-23 23:20:03 +01:00
tevent_req_set_callback ( subreq , rpc_smbd_write_done , req ) ;
return req ;
2009-01-29 22:54:55 +01:00
fail :
2009-03-23 23:20:03 +01:00
TALLOC_FREE ( req ) ;
2009-01-29 22:54:55 +01:00
return NULL ;
}
2009-03-23 23:20:03 +01:00
static void rpc_smbd_write_done ( struct tevent_req * subreq )
2009-01-29 22:54:55 +01:00
{
2009-03-23 23:20:03 +01:00
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct rpc_smbd_write_state * state = tevent_req_data (
req , struct rpc_smbd_write_state ) ;
2009-01-29 22:54:55 +01:00
NTSTATUS status ;
2010-03-25 15:00:38 +01:00
status = state - > transp - > sub_transp - > write_recv ( subreq , & state - > written ) ;
2009-01-29 22:54:55 +01:00
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2010-03-25 15:00:38 +01:00
rpc_smbd_disconnect ( state - > transp ) ;
2009-03-23 23:20:03 +01:00
tevent_req_nterror ( req , status ) ;
2009-01-29 22:54:55 +01:00
return ;
}
2009-03-23 23:20:03 +01:00
tevent_req_done ( req ) ;
2009-01-29 22:54:55 +01:00
}
2009-03-23 23:20:03 +01:00
static NTSTATUS rpc_smbd_write_recv ( struct tevent_req * req , ssize_t * pwritten )
2009-01-29 22:54:55 +01:00
{
2009-03-23 23:20:03 +01:00
struct rpc_smbd_write_state * state = tevent_req_data (
req , struct rpc_smbd_write_state ) ;
2009-01-29 22:54:55 +01:00
NTSTATUS status ;
2009-03-23 23:20:03 +01:00
if ( tevent_req_is_nterror ( req , & status ) ) {
2009-01-29 22:54:55 +01:00
return status ;
}
* pwritten = state - > written ;
return NT_STATUS_OK ;
}
struct rpc_smbd_read_state {
2010-03-25 15:00:38 +01:00
struct rpc_transport_smbd_state * transp ;
2009-01-29 22:54:55 +01:00
ssize_t received ;
} ;
2009-03-23 23:03:37 +01:00
static void rpc_smbd_read_done ( struct tevent_req * subreq ) ;
2009-01-29 22:54:55 +01:00
2009-03-23 23:03:37 +01:00
static struct tevent_req * rpc_smbd_read_send ( TALLOC_CTX * mem_ctx ,
struct event_context * ev ,
uint8_t * data , size_t size ,
void * priv )
2009-01-29 22:54:55 +01:00
{
struct rpc_transport_smbd_state * transp = talloc_get_type_abort (
priv , struct rpc_transport_smbd_state ) ;
2009-03-23 23:03:37 +01:00
struct tevent_req * req , * subreq ;
2009-01-29 22:54:55 +01:00
struct rpc_smbd_read_state * state ;
2010-03-25 15:00:38 +01:00
bool ok ;
2009-01-29 22:54:55 +01:00
2009-03-23 23:03:37 +01:00
req = tevent_req_create ( mem_ctx , & state , struct rpc_smbd_read_state ) ;
if ( req = = NULL ) {
2009-01-29 22:54:55 +01:00
return NULL ;
}
2010-03-25 15:00:38 +01:00
ok = rpc_smbd_is_connected ( transp ) ;
if ( ! ok ) {
tevent_req_nterror ( req , NT_STATUS_CONNECTION_INVALID ) ;
return tevent_req_post ( req , ev ) ;
}
state - > transp = transp ;
2009-01-29 22:54:55 +01:00
subreq = transp - > sub_transp - > read_send ( state , ev , data , size ,
transp - > sub_transp - > priv ) ;
if ( subreq = = NULL ) {
goto fail ;
}
2009-02-12 12:12:04 +01:00
if ( event_add_fd ( ev , state , transp - > conn - > stdout_fd , EVENT_FD_READ ,
2009-01-29 22:54:55 +01:00
rpc_cli_smbd_stdout_reader , transp - > conn ) = = NULL ) {
goto fail ;
}
2009-03-23 23:03:37 +01:00
tevent_req_set_callback ( subreq , rpc_smbd_read_done , req ) ;
return req ;
2009-01-29 22:54:55 +01:00
fail :
2009-03-23 23:03:37 +01:00
TALLOC_FREE ( req ) ;
2009-01-29 22:54:55 +01:00
return NULL ;
}
2009-03-23 23:03:37 +01:00
static void rpc_smbd_read_done ( struct tevent_req * subreq )
2009-01-29 22:54:55 +01:00
{
2009-03-23 23:03:37 +01:00
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct rpc_smbd_read_state * state = tevent_req_data (
req , struct rpc_smbd_read_state ) ;
2009-01-29 22:54:55 +01:00
NTSTATUS status ;
2010-03-25 15:00:38 +01:00
status = state - > transp - > sub_transp - > read_recv ( subreq , & state - > received ) ;
2009-01-29 22:54:55 +01:00
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2010-03-25 15:00:38 +01:00
rpc_smbd_disconnect ( state - > transp ) ;
2009-03-23 23:03:37 +01:00
tevent_req_nterror ( req , status ) ;
2009-01-29 22:54:55 +01:00
return ;
}
2009-03-23 23:03:37 +01:00
tevent_req_done ( req ) ;
2009-01-29 22:54:55 +01:00
}
2009-03-23 23:03:37 +01:00
static NTSTATUS rpc_smbd_read_recv ( struct tevent_req * req , ssize_t * preceived )
2009-01-29 22:54:55 +01:00
{
2009-03-23 23:03:37 +01:00
struct rpc_smbd_read_state * state = tevent_req_data (
req , struct rpc_smbd_read_state ) ;
2009-01-29 22:54:55 +01:00
NTSTATUS status ;
2009-03-23 23:03:37 +01:00
if ( tevent_req_is_nterror ( req , & status ) ) {
2009-01-29 22:54:55 +01:00
return status ;
}
* preceived = state - > received ;
return NT_STATUS_OK ;
}
struct rpc_transport_smbd_init_state {
struct rpc_cli_transport * transport ;
struct rpc_transport_smbd_state * transport_smbd ;
} ;
2009-04-08 21:33:01 +02:00
static void rpc_transport_smbd_init_done ( struct tevent_req * subreq ) ;
2009-01-29 22:54:55 +01:00
2009-04-08 21:58:41 +02:00
struct tevent_req * rpc_transport_smbd_init_send ( TALLOC_CTX * mem_ctx ,
struct event_context * ev ,
struct rpc_cli_smbd_conn * conn ,
const struct ndr_syntax_id * abstract_syntax )
2009-01-29 22:54:55 +01:00
{
2009-04-08 21:58:41 +02:00
struct tevent_req * req , * subreq ;
2009-01-29 22:54:55 +01:00
struct rpc_transport_smbd_init_state * state ;
2009-04-08 21:58:41 +02:00
req = tevent_req_create ( mem_ctx , & state ,
struct rpc_transport_smbd_init_state ) ;
if ( req = = NULL ) {
2009-01-29 22:54:55 +01:00
return NULL ;
}
state - > transport = talloc ( state , struct rpc_cli_transport ) ;
2009-04-08 21:58:41 +02:00
if ( tevent_req_nomem ( state - > transport , req ) ) {
return tevent_req_post ( req , ev ) ;
2009-01-29 22:54:55 +01:00
}
state - > transport_smbd = talloc ( state - > transport ,
struct rpc_transport_smbd_state ) ;
2009-04-08 21:58:41 +02:00
if ( tevent_req_nomem ( state - > transport_smbd , req ) ) {
return tevent_req_post ( req , ev ) ;
2009-01-29 22:54:55 +01:00
}
state - > transport_smbd - > conn = conn ;
state - > transport - > priv = state - > transport_smbd ;
2009-02-12 12:12:04 +01:00
if ( event_add_fd ( ev , state , conn - > stdout_fd , EVENT_FD_READ ,
rpc_cli_smbd_stdout_reader , conn ) = = NULL ) {
2009-04-08 21:58:41 +02:00
tevent_req_nterror ( req , NT_STATUS_NO_MEMORY ) ;
return tevent_req_post ( req , ev ) ;
2009-02-12 12:12:04 +01:00
}
2009-01-29 22:54:55 +01:00
subreq = rpc_transport_np_init_send ( state , ev , conn - > cli ,
abstract_syntax ) ;
2009-04-08 21:58:41 +02:00
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
2009-01-29 22:54:55 +01:00
}
2009-04-08 21:58:41 +02:00
tevent_req_set_callback ( subreq , rpc_transport_smbd_init_done , req ) ;
return req ;
2009-01-29 22:54:55 +01:00
}
2009-04-08 21:33:01 +02:00
static void rpc_transport_smbd_init_done ( struct tevent_req * subreq )
2009-01-29 22:54:55 +01:00
{
2009-04-08 21:58:41 +02:00
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct rpc_transport_smbd_init_state * state = tevent_req_data (
req , struct rpc_transport_smbd_init_state ) ;
2009-01-29 22:54:55 +01:00
NTSTATUS status ;
status = rpc_transport_np_init_recv (
subreq , state - > transport_smbd ,
& state - > transport_smbd - > sub_transp ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-04-08 21:58:41 +02:00
tevent_req_nterror ( req , status ) ;
2009-01-29 22:54:55 +01:00
return ;
}
2009-04-08 21:58:41 +02:00
tevent_req_done ( req ) ;
2009-01-29 22:54:55 +01:00
}
2009-04-08 21:58:41 +02:00
NTSTATUS rpc_transport_smbd_init_recv ( struct tevent_req * req ,
2009-01-29 22:54:55 +01:00
TALLOC_CTX * mem_ctx ,
struct rpc_cli_transport * * presult )
{
2009-04-08 21:58:41 +02:00
struct rpc_transport_smbd_init_state * state = tevent_req_data (
req , struct rpc_transport_smbd_init_state ) ;
2009-01-29 22:54:55 +01:00
NTSTATUS status ;
2009-04-08 21:58:41 +02:00
if ( tevent_req_is_nterror ( req , & status ) ) {
2009-01-29 22:54:55 +01:00
return status ;
}
state - > transport - > write_send = rpc_smbd_write_send ;
state - > transport - > write_recv = rpc_smbd_write_recv ;
state - > transport - > read_send = rpc_smbd_read_send ;
state - > transport - > read_recv = rpc_smbd_read_recv ;
state - > transport - > trans_send = NULL ;
state - > transport - > trans_recv = NULL ;
2010-03-25 15:00:38 +01:00
state - > transport - > is_connected = rpc_smbd_is_connected ;
2010-03-25 15:51:51 +01:00
state - > transport - > set_timeout = rpc_smbd_set_timeout ;
2009-01-29 22:54:55 +01:00
* presult = talloc_move ( mem_ctx , & state - > transport ) ;
return NT_STATUS_OK ;
}
NTSTATUS rpc_transport_smbd_init ( TALLOC_CTX * mem_ctx ,
struct rpc_cli_smbd_conn * conn ,
const struct ndr_syntax_id * abstract_syntax ,
struct rpc_cli_transport * * presult )
{
TALLOC_CTX * frame = talloc_stackframe ( ) ;
struct event_context * ev ;
2009-04-08 21:58:41 +02:00
struct tevent_req * req ;
2009-01-29 22:54:55 +01:00
NTSTATUS status ;
ev = event_context_init ( frame ) ;
if ( ev = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
req = rpc_transport_smbd_init_send ( frame , ev , conn , abstract_syntax ) ;
if ( req = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
2009-04-08 21:58:41 +02:00
if ( ! tevent_req_poll ( req , ev ) ) {
status = map_nt_error_from_unix ( errno ) ;
goto fail ;
2009-01-29 22:54:55 +01:00
}
status = rpc_transport_smbd_init_recv ( req , mem_ctx , presult ) ;
fail :
TALLOC_FREE ( frame ) ;
return status ;
}