2011-05-05 18:12:07 +02:00
/*
Unix SMB / CIFS implementation .
smb2 lib
Copyright ( C ) Volker Lendecke 2011
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"
# include "client.h"
# include "async_smb.h"
2011-09-07 18:44:21 +02:00
# include "../libcli/smb/smbXcli_base.h"
2011-05-05 18:12:07 +02:00
# include "smb2cli.h"
# include "libsmb/proto.h"
# include "lib/util/tevent_ntstatus.h"
struct smb2cli_write_state {
uint8_t fixed [ 48 ] ;
2011-09-05 18:22:57 +02:00
uint8_t dyn_pad [ 1 ] ;
2011-05-05 18:12:07 +02:00
} ;
static void smb2cli_write_done ( struct tevent_req * subreq ) ;
struct tevent_req * smb2cli_write_send ( TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
struct cli_state * cli ,
uint32_t length ,
uint64_t offset ,
uint64_t fid_persistent ,
uint64_t fid_volatile ,
uint32_t remaining_bytes ,
uint32_t flags ,
const uint8_t * data )
{
struct tevent_req * req , * subreq ;
struct smb2cli_write_state * state ;
uint8_t * fixed ;
2011-09-05 18:22:57 +02:00
const uint8_t * dyn ;
size_t dyn_len ;
2011-05-05 18:12:07 +02:00
req = tevent_req_create ( mem_ctx , & state ,
struct smb2cli_write_state ) ;
if ( req = = NULL ) {
return NULL ;
}
fixed = state - > fixed ;
SSVAL ( fixed , 0 , 49 ) ;
SSVAL ( fixed , 2 , SMB2_HDR_BODY + 48 ) ;
SIVAL ( fixed , 4 , length ) ;
SBVAL ( fixed , 8 , offset ) ;
SBVAL ( fixed , 16 , fid_persistent ) ;
SBVAL ( fixed , 24 , fid_volatile ) ;
SIVAL ( fixed , 36 , remaining_bytes ) ;
SIVAL ( fixed , 44 , flags ) ;
2011-09-05 18:22:57 +02:00
if ( length > 0 ) {
dyn = data ;
dyn_len = length ;
} else {
dyn = state - > dyn_pad ; ;
dyn_len = sizeof ( state - > dyn_pad ) ;
}
2011-09-07 18:44:21 +02:00
subreq = smb2cli_req_send ( state , ev , cli - > conn , SMB2_OP_WRITE ,
2011-08-12 17:26:13 +02:00
0 , 0 , /* flags */
2011-09-17 19:56:50 +02:00
cli - > timeout ,
2011-08-12 17:26:13 +02:00
cli - > smb2 . pid ,
cli - > smb2 . tid ,
2011-09-03 10:18:17 +02:00
cli - > smb2 . session ,
2011-05-05 18:12:07 +02:00
state - > fixed , sizeof ( state - > fixed ) ,
2011-09-05 18:22:57 +02:00
dyn , dyn_len ) ;
2011-05-05 18:12:07 +02:00
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
}
tevent_req_set_callback ( subreq , smb2cli_write_done , req ) ;
return req ;
}
static void smb2cli_write_done ( struct tevent_req * subreq )
{
struct tevent_req * req =
tevent_req_callback_data ( subreq ,
struct tevent_req ) ;
NTSTATUS status ;
struct iovec * iov ;
2011-08-31 00:40:06 +02:00
static const struct smb2cli_req_expected_response expected [ ] = {
{
. status = NT_STATUS_OK ,
. body_size = 0x11
}
} ;
2011-05-05 18:12:07 +02:00
2011-08-31 00:40:06 +02:00
status = smb2cli_req_recv ( subreq , talloc_tos ( ) , & iov ,
expected , ARRAY_SIZE ( expected ) ) ;
2011-05-05 18:12:07 +02:00
if ( tevent_req_nterror ( req , status ) ) {
return ;
}
tevent_req_done ( req ) ;
}
NTSTATUS smb2cli_write_recv ( struct tevent_req * req )
{
return tevent_req_simple_recv_ntstatus ( req ) ;
}
NTSTATUS smb2cli_write ( struct cli_state * cli ,
uint32_t length ,
uint64_t offset ,
uint64_t fid_persistent ,
uint64_t fid_volatile ,
uint32_t remaining_bytes ,
uint32_t flags ,
const uint8_t * data )
{
TALLOC_CTX * frame = talloc_stackframe ( ) ;
struct event_context * ev ;
struct tevent_req * req ;
NTSTATUS status = NT_STATUS_NO_MEMORY ;
if ( cli_has_async_calls ( cli ) ) {
/*
* Can ' t use sync call while an async call is in flight
*/
status = NT_STATUS_INVALID_PARAMETER ;
goto fail ;
}
ev = event_context_init ( frame ) ;
if ( ev = = NULL ) {
goto fail ;
}
req = smb2cli_write_send ( frame , ev , cli , length , offset ,
fid_persistent , fid_volatile ,
remaining_bytes , flags , data ) ;
if ( req = = NULL ) {
goto fail ;
}
if ( ! tevent_req_poll_ntstatus ( req , ev , & status ) ) {
goto fail ;
}
status = smb2cli_write_recv ( req ) ;
fail :
TALLOC_FREE ( frame ) ;
return status ;
}