2012-03-03 01:23:44 +01:00
/*
2002-01-30 06:08:46 +00:00
Unix SMB / CIFS implementation .
1996-08-15 15:11:34 +00:00
Pipe SMB reply routines
1998-01-22 13:27:43 +00:00
Copyright ( C ) Andrew Tridgell 1992 - 1998
Copyright ( C ) Luke Kenneth Casson Leighton 1996 - 1998
Copyright ( C ) Paul Ashton 1997 - 1998.
2005-07-08 04:51:27 +00:00
Copyright ( C ) Jeremy Allison 2005.
2012-03-03 01:23:44 +01:00
1996-08-15 15:11:34 +00: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
2007-07-09 19:25:36 +00:00
the Free Software Foundation ; either version 3 of the License , or
1996-08-15 15:11:34 +00:00
( at your option ) any later version .
2012-03-03 01:23:44 +01:00
1996-08-15 15:11:34 +00:00
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 .
2012-03-03 01:23:44 +01:00
1996-08-15 15:11:34 +00:00
You should have received a copy of the GNU General Public License
2007-07-10 00:52:41 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
1996-08-15 15:11:34 +00:00
*/
/*
This file handles reply_ calls on named pipes that the server
makes to handle specific protocols
*/
# include "includes.h"
2011-03-22 16:57:01 +01:00
# include "smbd/smbd.h"
2010-04-27 15:12:32 +02:00
# include "smbd/globals.h"
2010-10-12 15:27:50 +11:00
# include "libcli/security/security.h"
2011-04-29 23:32:28 +02:00
# include "rpc_server/srv_pipe_hnd.h"
1996-08-15 15:11:34 +00:00
2009-01-20 15:21:04 +01:00
NTSTATUS open_np_file ( struct smb_request * smb_req , const char * name ,
struct files_struct * * pfsp )
{
struct connection_struct * conn = smb_req - > conn ;
struct files_struct * fsp ;
2009-07-10 14:50:37 -07:00
struct smb_filename * smb_fname = NULL ;
2009-01-20 15:21:04 +01:00
NTSTATUS status ;
status = file_new ( smb_req , conn , & fsp ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " file_new failed: %s \n " , nt_errstr ( status ) ) ) ;
return status ;
}
fsp - > conn = conn ;
fsp - > fh - > fd = - 1 ;
fsp - > vuid = smb_req - > vuid ;
fsp - > can_lock = false ;
fsp - > access_mask = FILE_READ_DATA | FILE_WRITE_DATA ;
2009-07-10 14:50:37 -07:00
2016-03-18 21:19:38 -07:00
smb_fname = synthetic_smb_fname ( talloc_tos ( ) , name , NULL , NULL , 0 ) ;
2013-04-15 11:00:14 +02:00
if ( smb_fname = = NULL ) {
2009-07-10 14:50:37 -07:00
file_free ( smb_req , fsp ) ;
2013-04-15 11:00:14 +02:00
return NT_STATUS_NO_MEMORY ;
2009-07-10 14:50:37 -07:00
}
status = fsp_set_smb_fname ( fsp , smb_fname ) ;
TALLOC_FREE ( smb_fname ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
file_free ( smb_req , fsp ) ;
return status ;
}
2009-01-20 15:21:04 +01:00
2010-04-27 15:12:32 +02:00
status = np_open ( fsp , name ,
conn - > sconn - > local_address ,
conn - > sconn - > remote_address ,
2011-02-21 10:25:52 +01:00
conn - > session_info ,
2013-10-23 17:03:37 +02:00
conn - > sconn - > ev_ctx ,
2010-08-08 09:11:45 +02:00
conn - > sconn - > msg_ctx ,
& fsp - > fake_file_handle ) ;
2009-01-20 15:21:04 +01:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 10 , ( " np_open(%s) returned %s \n " , name ,
nt_errstr ( status ) ) ) ;
file_free ( smb_req , fsp ) ;
return status ;
}
* pfsp = fsp ;
return NT_STATUS_OK ;
}
1997-10-30 01:05:13 +00:00
/****************************************************************************
2005-07-08 04:51:27 +00:00
Reply to an open and X on a named pipe .
This code is basically stolen from reply_open_and_X with some
wrinkles to handle pipes .
1996-08-15 15:11:34 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-07-08 04:51:27 +00:00
2007-07-23 09:54:36 +00:00
void reply_open_pipe_and_X ( connection_struct * conn , struct smb_request * req )
1996-08-15 15:11:34 +00:00
{
2007-09-12 23:50:21 +00:00
const char * fname = NULL ;
char * pipe_name = NULL ;
2008-10-12 23:01:38 +02:00
files_struct * fsp ;
2007-09-12 23:50:21 +00:00
TALLOC_CTX * ctx = talloc_tos ( ) ;
2008-10-12 23:01:38 +02:00
NTSTATUS status ;
1998-08-17 03:06:20 +00:00
/* XXXX we need to handle passed times, sattr and flags */
2008-11-02 01:07:46 +01:00
srvstr_pull_req_talloc ( ctx , req , & pipe_name , req - > buf , STR_TERMINATE ) ;
2007-09-12 23:50:21 +00:00
if ( ! pipe_name ) {
reply_botherror ( req , NT_STATUS_OBJECT_NAME_NOT_FOUND ,
ERRDOS , ERRbadpipe ) ;
return ;
}
1998-08-17 03:06:20 +00:00
/* If the name doesn't start \PIPE\ then this is directed */
/* at a mailslot or something we really, really don't understand, */
/* not just something we really don't understand. */
2012-05-28 18:23:50 +02:00
# define PIPE "PIPE\\"
# define PIPELEN strlen(PIPE)
fname = pipe_name ;
while ( fname [ 0 ] = = ' \\ ' ) {
fname + + ;
}
if ( ! strnequal ( fname , PIPE , PIPELEN ) ) {
reply_nterror ( req , NT_STATUS_OBJECT_PATH_SYNTAX_BAD ) ;
2007-07-23 09:54:36 +00:00
return ;
2005-07-08 04:51:27 +00:00
}
2012-05-28 18:23:50 +02:00
fname + = PIPELEN ;
while ( fname [ 0 ] = = ' \\ ' ) {
fname + + ;
}
1998-08-17 03:06:20 +00:00
2012-05-28 18:23:50 +02:00
DEBUG ( 4 , ( " Opening pipe %s => %s. \n " , pipe_name , fname ) ) ;
2001-07-04 21:57:03 +00:00
2000-06-07 01:49:23 +00:00
#if 0
1999-12-13 13:27:58 +00:00
/*
* Hack for NT printers . . . JRA .
*/
2007-07-23 09:54:36 +00:00
if ( should_fail_next_srvsvc_open ( fname ) ) {
2009-12-21 11:05:25 -08:00
reply_nterror ( req , NT_STATUS_ACCESS_DENIED ) ;
2007-07-23 09:54:36 +00:00
return ;
}
2000-06-07 01:49:23 +00:00
# endif
1999-12-13 13:27:58 +00:00
2009-01-20 15:21:04 +01:00
status = open_np_file ( req , fname , & fsp ) ;
2008-10-12 23:01:38 +02:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2008-10-25 15:23:36 +02:00
if ( NT_STATUS_EQUAL ( status , NT_STATUS_OBJECT_NAME_NOT_FOUND ) ) {
reply_botherror ( req , NT_STATUS_OBJECT_NAME_NOT_FOUND ,
ERRDOS , ERRbadpipe ) ;
return ;
}
2008-10-12 23:01:38 +02:00
reply_nterror ( req , status ) ;
2007-07-23 09:54:36 +00:00
return ;
2005-07-08 04:51:27 +00:00
}
1998-08-17 03:06:20 +00:00
/* Prepare the reply */
2007-07-23 09:54:36 +00:00
reply_outbuf ( req , 15 , 0 ) ;
1998-08-17 03:06:20 +00:00
2012-02-26 16:49:09 +01:00
SSVAL ( req - > outbuf , smb_vwv0 , 0xff ) ; /* andx chain ends */
SSVAL ( req - > outbuf , smb_vwv1 , 0 ) ; /* no andx offset */
1998-08-17 03:06:20 +00:00
/* Mark the opened file as an existing named pipe in message mode. */
2007-07-23 09:54:36 +00:00
SSVAL ( req - > outbuf , smb_vwv9 , 2 ) ;
SSVAL ( req - > outbuf , smb_vwv10 , 0xc700 ) ;
1998-08-17 03:06:20 +00:00
2008-11-09 19:57:10 +01:00
SSVAL ( req - > outbuf , smb_vwv2 , fsp - > fnum ) ;
SSVAL ( req - > outbuf , smb_vwv3 , 0 ) ; /* fmode */
srv_put_dos_date3 ( ( char * ) req - > outbuf , smb_vwv4 , 0 ) ; /* mtime */
SIVAL ( req - > outbuf , smb_vwv6 , 0 ) ; /* size */
SSVAL ( req - > outbuf , smb_vwv8 , 0 ) ; /* rmode */
SSVAL ( req - > outbuf , smb_vwv11 , 0x0001 ) ;
1996-08-15 15:11:34 +00:00
}
1999-11-15 22:11:10 +00:00
/****************************************************************************
2005-07-08 04:51:27 +00:00
Reply to a write on a pipe .
1999-11-15 22:11:10 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-07-08 04:51:27 +00:00
2009-01-31 14:43:06 +01:00
struct pipe_write_state {
size_t numtowrite ;
} ;
2009-03-17 09:17:16 +01:00
static void pipe_write_done ( struct tevent_req * subreq ) ;
2009-01-31 14:43:06 +01:00
2007-08-14 18:16:04 +00:00
void reply_pipe_write ( struct smb_request * req )
1999-11-15 22:11:10 +00:00
{
2008-11-02 12:20:47 +01:00
files_struct * fsp = file_fsp ( req , SVAL ( req - > vwv + 0 , 0 ) ) ;
2008-11-01 17:35:48 +01:00
const uint8_t * data ;
2009-01-31 14:43:06 +01:00
struct pipe_write_state * state ;
2009-03-17 09:17:16 +01:00
struct tevent_req * subreq ;
1999-11-15 22:11:10 +00:00
2008-10-12 23:01:38 +02:00
if ( ! fsp_is_np ( fsp ) ) {
2009-12-21 11:05:25 -08:00
reply_nterror ( req , NT_STATUS_INVALID_HANDLE ) ;
2007-08-14 18:16:04 +00:00
return ;
2005-07-08 04:51:27 +00:00
}
1999-11-15 22:11:10 +00:00
2008-10-12 23:01:38 +02:00
if ( fsp - > vuid ! = req - > vuid ) {
2007-08-14 18:16:04 +00:00
reply_nterror ( req , NT_STATUS_INVALID_HANDLE ) ;
return ;
2006-05-27 21:38:54 +00:00
}
2009-01-31 14:43:06 +01:00
state = talloc ( req , struct pipe_write_state ) ;
if ( state = = NULL ) {
reply_nterror ( req , NT_STATUS_NO_MEMORY ) ;
return ;
}
req - > async_priv = state ;
state - > numtowrite = SVAL ( req - > vwv + 1 , 0 ) ;
2008-11-01 17:35:48 +01:00
data = req - > buf + 3 ;
1999-11-15 22:11:10 +00:00
2012-06-14 12:57:25 +02:00
DEBUG ( 6 , ( " reply_pipe_write: %s, name: %s len: %d \n " , fsp_fnum_dbg ( fsp ) ,
2009-07-10 14:50:37 -07:00
fsp_str_dbg ( fsp ) , ( int ) state - > numtowrite ) ) ;
2009-01-31 14:43:06 +01:00
2011-12-12 13:47:56 +01:00
subreq = np_write_send ( state , req - > sconn - > ev_ctx ,
2009-01-31 14:43:06 +01:00
fsp - > fake_file_handle , data , state - > numtowrite ) ;
if ( subreq = = NULL ) {
TALLOC_FREE ( state ) ;
reply_nterror ( req , NT_STATUS_NO_MEMORY ) ;
2009-01-31 14:33:38 +01:00
return ;
2005-07-08 04:51:27 +00:00
}
2009-03-17 09:17:16 +01:00
tevent_req_set_callback ( subreq , pipe_write_done ,
talloc_move ( req - > conn , & req ) ) ;
2009-01-31 14:43:06 +01:00
}
2009-03-17 09:17:16 +01:00
static void pipe_write_done ( struct tevent_req * subreq )
2009-01-31 14:43:06 +01:00
{
2009-03-17 09:17:16 +01:00
struct smb_request * req = tevent_req_callback_data (
subreq , struct smb_request ) ;
2009-01-31 14:43:06 +01:00
struct pipe_write_state * state = talloc_get_type_abort (
req - > async_priv , struct pipe_write_state ) ;
NTSTATUS status ;
ssize_t nwritten = - 1 ;
1999-11-15 22:11:10 +00:00
2009-01-31 14:43:06 +01:00
status = np_write_recv ( subreq , & nwritten ) ;
TALLOC_FREE ( subreq ) ;
2009-07-10 18:23:00 -07:00
if ( nwritten < 0 ) {
reply_nterror ( req , status ) ;
goto send ;
}
/* Looks bogus to me now. Needs to be removed ? JRA. */
if ( ( nwritten = = 0 & & state - > numtowrite ! = 0 ) ) {
2009-12-21 11:05:25 -08:00
reply_nterror ( req , NT_STATUS_ACCESS_DENIED ) ;
2009-01-31 14:43:06 +01:00
goto send ;
2005-07-08 04:51:27 +00:00
}
1999-11-15 22:11:10 +00:00
2007-08-14 18:16:04 +00:00
reply_outbuf ( req , 1 , 0 ) ;
SSVAL ( req - > outbuf , smb_vwv0 , nwritten ) ;
1999-11-15 22:11:10 +00:00
2009-01-31 14:43:06 +01:00
DEBUG ( 3 , ( " write-IPC nwritten=%d \n " , ( int ) nwritten ) ) ;
send :
2014-06-11 12:55:24 +02:00
if ( ! srv_send_smb ( req - > xconn , ( char * ) req - > outbuf ,
2009-03-09 09:47:59 +01:00
true , req - > seqnum + 1 ,
2009-02-08 23:10:34 -08:00
IS_CONN_ENCRYPTED ( req - > conn ) | | req - > encrypted ,
& req - > pcd ) ) {
2009-01-31 14:43:06 +01:00
exit_server_cleanly ( " construct_reply: srv_send_smb failed. " ) ;
}
TALLOC_FREE ( req ) ;
1999-11-15 22:11:10 +00:00
}
1996-08-15 15:11:34 +00:00
1998-10-07 15:22:49 +00:00
/****************************************************************************
2000-03-09 21:45:16 +00:00
Reply to a write and X .
1998-10-07 15:22:49 +00:00
2000-03-09 21:45:16 +00:00
This code is basically stolen from reply_write_and_X with some
wrinkles to handle pipes .
1998-10-07 15:22:49 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-03-09 21:45:16 +00:00
2009-01-31 14:31:31 +01:00
struct pipe_write_andx_state {
bool pipe_start_message_raw ;
size_t numtowrite ;
} ;
2009-03-17 09:17:16 +01:00
static void pipe_write_andx_done ( struct tevent_req * subreq ) ;
2009-01-31 14:31:31 +01:00
2007-08-08 18:40:26 +00:00
void reply_pipe_write_and_X ( struct smb_request * req )
1998-10-07 15:22:49 +00:00
{
2008-11-02 12:20:47 +01:00
files_struct * fsp = file_fsp ( req , SVAL ( req - > vwv + 2 , 0 ) ) ;
int smb_doff = SVAL ( req - > vwv + 11 , 0 ) ;
2011-05-05 10:41:59 -07:00
const uint8_t * data ;
2009-01-31 14:31:31 +01:00
struct pipe_write_andx_state * state ;
2009-03-17 09:17:16 +01:00
struct tevent_req * subreq ;
1998-10-07 15:22:49 +00:00
2008-10-12 23:01:38 +02:00
if ( ! fsp_is_np ( fsp ) ) {
2009-12-21 11:05:25 -08:00
reply_nterror ( req , NT_STATUS_INVALID_HANDLE ) ;
2007-08-08 18:40:26 +00:00
return ;
2005-07-08 04:51:27 +00:00
}
1998-10-07 15:22:49 +00:00
2008-10-12 23:01:38 +02:00
if ( fsp - > vuid ! = req - > vuid ) {
2007-08-08 18:40:26 +00:00
reply_nterror ( req , NT_STATUS_INVALID_HANDLE ) ;
return ;
2006-05-27 21:38:54 +00:00
}
2009-01-31 14:31:31 +01:00
state = talloc ( req , struct pipe_write_andx_state ) ;
if ( state = = NULL ) {
reply_nterror ( req , NT_STATUS_NO_MEMORY ) ;
return ;
}
req - > async_priv = state ;
state - > numtowrite = SVAL ( req - > vwv + 10 , 0 ) ;
state - > pipe_start_message_raw =
( ( SVAL ( req - > vwv + 7 , 0 ) & ( PIPE_START_MESSAGE | PIPE_RAW_MODE ) )
= = ( PIPE_START_MESSAGE | PIPE_RAW_MODE ) ) ;
2012-06-14 12:57:25 +02:00
DEBUG ( 6 , ( " reply_pipe_write_and_X: %s, name: %s len: %d \n " ,
fsp_fnum_dbg ( fsp ) , fsp_str_dbg ( fsp ) , ( int ) state - > numtowrite ) ) ;
2009-01-20 15:21:04 +01:00
2011-05-05 10:41:59 -07:00
data = ( const uint8_t * ) smb_base ( req - > inbuf ) + smb_doff ;
1998-10-07 15:22:49 +00:00
2009-01-31 14:31:31 +01:00
if ( state - > pipe_start_message_raw ) {
2009-01-31 14:33:38 +01:00
/*
* For the start of a message in named pipe byte mode ,
* the first two bytes are a length - of - pdu field . Ignore
* them ( we don ' t trust the client ) . JRA .
*/
2009-01-31 14:31:31 +01:00
if ( state - > numtowrite < 2 ) {
2009-01-31 14:33:38 +01:00
DEBUG ( 0 , ( " reply_pipe_write_and_X: start of message "
" set and not enough data sent.(%u) \n " ,
2009-01-31 14:31:31 +01:00
( unsigned int ) state - > numtowrite ) ) ;
2009-07-10 18:23:00 -07:00
reply_nterror ( req , NT_STATUS_INVALID_PARAMETER ) ;
2008-10-12 23:01:38 +02:00
return ;
}
2009-01-31 14:33:38 +01:00
data + = 2 ;
2009-01-31 14:31:31 +01:00
state - > numtowrite - = 2 ;
2009-01-31 14:33:38 +01:00
}
2009-01-31 14:31:31 +01:00
2011-12-12 13:47:56 +01:00
subreq = np_write_send ( state , req - > sconn - > ev_ctx ,
2009-01-31 14:31:31 +01:00
fsp - > fake_file_handle , data , state - > numtowrite ) ;
if ( subreq = = NULL ) {
TALLOC_FREE ( state ) ;
reply_nterror ( req , NT_STATUS_NO_MEMORY ) ;
2009-01-31 14:33:38 +01:00
return ;
2000-03-09 21:45:16 +00:00
}
2009-03-17 09:17:16 +01:00
tevent_req_set_callback ( subreq , pipe_write_andx_done ,
talloc_move ( req - > conn , & req ) ) ;
2009-01-31 14:31:31 +01:00
}
1998-10-07 15:22:49 +00:00
2009-03-17 09:17:16 +01:00
static void pipe_write_andx_done ( struct tevent_req * subreq )
2009-01-31 14:31:31 +01:00
{
2009-03-17 09:17:16 +01:00
struct smb_request * req = tevent_req_callback_data (
subreq , struct smb_request ) ;
2009-01-31 14:31:31 +01:00
struct pipe_write_andx_state * state = talloc_get_type_abort (
req - > async_priv , struct pipe_write_andx_state ) ;
NTSTATUS status ;
ssize_t nwritten = - 1 ;
status = np_write_recv ( subreq , & nwritten ) ;
TALLOC_FREE ( subreq ) ;
2009-07-10 18:23:00 -07:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
reply_nterror ( req , status ) ;
goto done ;
}
/* Looks bogus to me now. Is this error message correct ? JRA. */
if ( nwritten ! = state - > numtowrite ) {
2009-12-21 11:05:25 -08:00
reply_nterror ( req , NT_STATUS_ACCESS_DENIED ) ;
2009-01-31 14:31:31 +01:00
goto done ;
2005-07-08 04:51:27 +00:00
}
2007-08-08 18:40:26 +00:00
reply_outbuf ( req , 6 , 0 ) ;
1998-10-07 15:22:49 +00:00
2012-02-26 16:49:09 +01:00
SSVAL ( req - > outbuf , smb_vwv0 , 0xff ) ; /* andx chain ends */
SSVAL ( req - > outbuf , smb_vwv1 , 0 ) ; /* no andx offset */
2009-01-31 14:31:31 +01:00
nwritten = ( state - > pipe_start_message_raw ? nwritten + 2 : nwritten ) ;
2007-08-08 18:40:26 +00:00
SSVAL ( req - > outbuf , smb_vwv2 , nwritten ) ;
1998-10-07 15:22:49 +00:00
2009-01-31 14:31:31 +01:00
DEBUG ( 3 , ( " writeX-IPC nwritten=%d \n " , ( int ) nwritten ) ) ;
done :
2010-01-05 14:34:00 -08:00
/*
* We must free here as the ownership of req was
* moved to the connection struct in reply_pipe_write_and_X ( ) .
*/
2012-03-04 12:10:04 +01:00
smb_request_done ( req ) ;
1998-10-07 15:22:49 +00:00
}
1997-10-30 01:05:13 +00:00
/****************************************************************************
2005-07-08 04:51:27 +00:00
Reply to a read and X .
This code is basically stolen from reply_read_and_X with some
wrinkles to handle pipes .
1996-08-15 15:11:34 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-07-08 04:51:27 +00:00
2009-01-31 14:57:52 +01:00
struct pipe_read_andx_state {
uint8_t * outbuf ;
int smb_mincnt ;
int smb_maxcnt ;
} ;
2009-03-17 09:34:17 +01:00
static void pipe_read_andx_done ( struct tevent_req * subreq ) ;
2009-01-31 14:57:52 +01:00
2007-08-11 10:53:36 +00:00
void reply_pipe_read_and_X ( struct smb_request * req )
1996-08-15 15:11:34 +00:00
{
2008-11-02 12:20:47 +01:00
files_struct * fsp = file_fsp ( req , SVAL ( req - > vwv + 0 , 0 ) ) ;
2008-10-12 23:01:38 +02:00
uint8_t * data ;
2009-01-31 14:57:52 +01:00
struct pipe_read_andx_state * state ;
2009-03-17 09:34:17 +01:00
struct tevent_req * subreq ;
2002-01-20 02:40:05 +00:00
1999-12-13 13:27:58 +00:00
/* we don't use the offset given to use for pipe reads. This
is deliberate , instead we always return the next lump of
data on the pipe */
#if 0
2015-05-02 21:01:14 -07:00
uint32_t smb_offs = IVAL ( req - > vwv + 3 , 0 ) ;
1999-12-13 13:27:58 +00:00
# endif
1996-08-15 15:11:34 +00:00
2008-10-12 23:01:38 +02:00
if ( ! fsp_is_np ( fsp ) ) {
2009-12-21 11:05:25 -08:00
reply_nterror ( req , NT_STATUS_INVALID_HANDLE ) ;
2007-08-11 10:53:36 +00:00
return ;
2005-07-08 04:51:27 +00:00
}
1998-09-01 01:10:01 +00:00
2008-10-12 23:01:38 +02:00
if ( fsp - > vuid ! = req - > vuid ) {
reply_nterror ( req , NT_STATUS_INVALID_HANDLE ) ;
return ;
}
2009-01-31 14:57:52 +01:00
state = talloc ( req , struct pipe_read_andx_state ) ;
if ( state = = NULL ) {
reply_nterror ( req , NT_STATUS_NO_MEMORY ) ;
return ;
}
req - > async_priv = state ;
2007-08-11 10:53:36 +00:00
2009-01-31 14:57:52 +01:00
state - > smb_maxcnt = SVAL ( req - > vwv + 5 , 0 ) ;
state - > smb_mincnt = SVAL ( req - > vwv + 6 , 0 ) ;
2014-08-14 22:04:33 -07:00
reply_outbuf ( req , 12 , state - > smb_maxcnt + 1 /* padding byte */ ) ;
2012-02-26 16:49:09 +01:00
SSVAL ( req - > outbuf , smb_vwv0 , 0xff ) ; /* andx chain ends */
SSVAL ( req - > outbuf , smb_vwv1 , 0 ) ; /* no andx offset */
2014-08-14 22:04:33 -07:00
SCVAL ( smb_buf ( req - > outbuf ) , 0 , 0 ) ; /* padding byte */
2012-02-26 16:49:09 +01:00
2014-08-14 22:04:33 -07:00
data = ( uint8_t * ) smb_buf ( req - > outbuf ) + 1 /* padding byte */ ;
1996-08-15 15:11:34 +00:00
2009-01-31 14:57:52 +01:00
/*
* We have to tell the upper layers that we ' re async .
*/
state - > outbuf = req - > outbuf ;
req - > outbuf = NULL ;
1996-08-15 15:11:34 +00:00
2011-12-12 13:47:56 +01:00
subreq = np_read_send ( state , req - > sconn - > ev_ctx ,
2009-01-31 14:57:52 +01:00
fsp - > fake_file_handle , data ,
state - > smb_maxcnt ) ;
if ( subreq = = NULL ) {
reply_nterror ( req , NT_STATUS_NO_MEMORY ) ;
2007-08-11 10:53:36 +00:00
return ;
2005-07-08 04:51:27 +00:00
}
2009-03-17 09:34:17 +01:00
tevent_req_set_callback ( subreq , pipe_read_andx_done ,
talloc_move ( req - > conn , & req ) ) ;
2009-01-31 14:57:52 +01:00
}
2009-03-17 09:34:17 +01:00
static void pipe_read_andx_done ( struct tevent_req * subreq )
2009-01-31 14:57:52 +01:00
{
2009-03-17 09:34:17 +01:00
struct smb_request * req = tevent_req_callback_data (
subreq , struct smb_request ) ;
2009-01-31 14:57:52 +01:00
struct pipe_read_andx_state * state = talloc_get_type_abort (
req - > async_priv , struct pipe_read_andx_state ) ;
NTSTATUS status ;
ssize_t nread ;
bool is_data_outstanding ;
status = np_read_recv ( subreq , & nread , & is_data_outstanding ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2012-05-23 15:23:23 +02:00
NTSTATUS old = status ;
status = nt_status_np_pipe ( old ) ;
2009-01-31 14:57:52 +01:00
reply_nterror ( req , status ) ;
goto done ;
}
req - > outbuf = state - > outbuf ;
state - > outbuf = NULL ;
2007-08-11 10:53:36 +00:00
2014-08-14 22:04:33 -07:00
srv_set_message ( ( char * ) req - > outbuf , 12 , nread + 1 /* padding byte */ ,
false ) ;
2010-04-28 14:36:10 +02:00
#if 0
/*
* we should return STATUS_BUFFER_OVERFLOW if there ' s
* out standing data .
*
* But we can ' t enable it yet , as it has bad interactions
* with fixup_chain_error_packet ( ) in chain_reply ( ) .
*/
if ( is_data_outstanding ) {
error_packet_set ( ( char * ) req - > outbuf , ERRDOS , ERRmoredata ,
STATUS_BUFFER_OVERFLOW , __LINE__ , __FILE__ ) ;
}
# endif
2007-08-11 10:53:36 +00:00
SSVAL ( req - > outbuf , smb_vwv5 , nread ) ;
2008-11-21 23:17:31 +01:00
SSVAL ( req - > outbuf , smb_vwv6 ,
2012-03-03 00:33:15 +01:00
( smb_wct - 4 ) /* offset from smb header to wct */
2008-11-21 23:17:31 +01:00
+ 1 /* the wct field */
+ 12 * sizeof ( uint16_t ) /* vwv */
2014-08-14 22:04:33 -07:00
+ 2 /* the buflen field */
+ 1 ) ; /* padding byte */
2012-03-03 01:23:44 +01:00
2009-01-31 14:57:52 +01:00
DEBUG ( 3 , ( " readX-IPC min=%d max=%d nread=%d \n " ,
state - > smb_mincnt , state - > smb_maxcnt , ( int ) nread ) ) ;
1996-08-15 15:11:34 +00:00
2009-01-31 14:57:52 +01:00
done :
2010-01-05 14:34:00 -08:00
/*
* We must free here as the ownership of req was
* moved to the connection struct in reply_pipe_read_and_X ( ) .
*/
2012-03-04 12:10:04 +01:00
smb_request_done ( req ) ;
1996-08-15 15:11:34 +00:00
}