2011-05-05 20:12:07 +04: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"
2012-05-15 12:50:49 +04:00
# include "system/network.h"
2011-05-05 20:12:07 +04:00
# include "lib/util/tevent_ntstatus.h"
2012-05-15 12:50:49 +04:00
# include "smb_common.h"
# include "smbXcli_base.h"
2011-05-05 20:12:07 +04:00
struct smb2cli_read_state {
uint8_t fixed [ 48 ] ;
2011-09-05 20:22:57 +04:00
uint8_t dyn_pad [ 1 ] ;
2011-07-08 19:57:56 +04:00
struct iovec * recv_iov ;
2011-05-05 20:12:07 +04:00
uint8_t * data ;
uint32_t data_length ;
2015-11-27 21:10:01 +03:00
bool out_valid ;
2011-05-05 20:12:07 +04:00
} ;
static void smb2cli_read_done ( struct tevent_req * subreq ) ;
struct tevent_req * smb2cli_read_send ( TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
2012-05-10 20:30:25 +04:00
struct smbXcli_conn * conn ,
uint32_t timeout_msec ,
struct smbXcli_session * session ,
2012-07-24 00:32:49 +04:00
struct smbXcli_tcon * tcon ,
2011-05-05 20:12:07 +04:00
uint32_t length ,
uint64_t offset ,
uint64_t fid_persistent ,
uint64_t fid_volatile ,
uint64_t minimum_count ,
uint64_t remaining_bytes )
{
struct tevent_req * req , * subreq ;
struct smb2cli_read_state * state ;
uint8_t * fixed ;
req = tevent_req_create ( mem_ctx , & state ,
struct smb2cli_read_state ) ;
if ( req = = NULL ) {
return NULL ;
}
fixed = state - > fixed ;
SSVAL ( fixed , 0 , 49 ) ;
SIVAL ( fixed , 4 , length ) ;
SBVAL ( fixed , 8 , offset ) ;
SBVAL ( fixed , 16 , fid_persistent ) ;
SBVAL ( fixed , 24 , fid_volatile ) ;
SBVAL ( fixed , 32 , minimum_count ) ;
SBVAL ( fixed , 40 , remaining_bytes ) ;
2012-05-10 20:30:25 +04:00
subreq = smb2cli_req_send ( state , ev , conn , SMB2_OP_READ ,
2011-08-12 19:26:13 +04:00
0 , 0 , /* flags */
2012-05-10 20:30:25 +04:00
timeout_msec ,
2012-07-25 12:36:27 +04:00
tcon ,
2012-05-10 20:30:25 +04:00
session ,
2011-05-05 20:12:07 +04:00
state - > fixed , sizeof ( state - > fixed ) ,
2013-08-13 12:25:52 +04:00
state - > dyn_pad , sizeof ( state - > dyn_pad ) ,
length ) ; /* max_dyn_len */
2011-05-05 20:12:07 +04:00
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
}
tevent_req_set_callback ( subreq , smb2cli_read_done , req ) ;
return req ;
}
static void smb2cli_read_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct smb2cli_read_state * state =
tevent_req_data ( req ,
struct smb2cli_read_state ) ;
NTSTATUS status ;
2021-06-29 16:42:56 +03:00
NTSTATUS error ;
2011-05-05 20:12:07 +04:00
struct iovec * iov ;
2021-06-29 16:42:56 +03:00
const uint8_t dyn_ofs = SMB2_HDR_BODY + 0x10 ;
DATA_BLOB dyn_buffer = data_blob_null ;
2011-05-05 20:12:07 +04:00
uint8_t data_offset ;
2021-06-29 16:42:56 +03:00
DATA_BLOB data_buffer = data_blob_null ;
uint32_t next_offset = 0 ; /* this variable is completely ignored */
2011-08-31 02:40:06 +04:00
static const struct smb2cli_req_expected_response expected [ ] = {
{
. status = STATUS_BUFFER_OVERFLOW ,
. body_size = 0x11
} ,
{
. status = NT_STATUS_OK ,
. body_size = 0x11
}
} ;
2011-05-05 20:12:07 +04:00
2011-08-31 02:40:06 +04:00
status = smb2cli_req_recv ( subreq , state , & iov ,
expected , ARRAY_SIZE ( expected ) ) ;
2012-09-28 21:48:26 +04:00
TALLOC_FREE ( subreq ) ;
2015-11-27 21:10:01 +03:00
if ( NT_STATUS_EQUAL ( status , STATUS_BUFFER_OVERFLOW ) ) {
/* no error */
} else {
if ( tevent_req_nterror ( req , status ) ) {
return ;
}
2011-05-05 20:12:07 +04:00
}
data_offset = CVAL ( iov [ 1 ] . iov_base , 2 ) ;
state - > data_length = IVAL ( iov [ 1 ] . iov_base , 4 ) ;
2021-06-29 16:42:56 +03:00
dyn_buffer = data_blob_const ( ( uint8_t * ) iov [ 2 ] . iov_base ,
iov [ 2 ] . iov_len ) ;
error = smb2cli_parse_dyn_buffer ( dyn_ofs ,
dyn_buffer ,
dyn_ofs , /* min_offset */
data_offset ,
state - > data_length ,
dyn_buffer . length , /* max_length */
& next_offset ,
& data_buffer ) ;
if ( tevent_req_nterror ( req , error ) ) {
2011-05-05 20:12:07 +04:00
return ;
}
2011-07-08 19:57:56 +04:00
state - > recv_iov = iov ;
2021-06-29 16:42:56 +03:00
state - > data = data_buffer . data ;
2015-11-27 21:10:01 +03:00
state - > out_valid = true ;
if ( tevent_req_nterror ( req , status ) ) {
return ;
}
2011-05-05 20:12:07 +04:00
tevent_req_done ( req ) ;
}
NTSTATUS smb2cli_read_recv ( struct tevent_req * req , TALLOC_CTX * mem_ctx ,
uint8_t * * data , uint32_t * data_length )
{
struct smb2cli_read_state * state =
tevent_req_data ( req ,
struct smb2cli_read_state ) ;
2015-11-27 21:10:01 +03:00
NTSTATUS status = NT_STATUS_OK ;
2011-05-05 20:12:07 +04:00
2015-11-27 21:10:01 +03:00
if ( tevent_req_is_nterror ( req , & status ) & & ! state - > out_valid ) {
* data_length = 0 ;
* data = NULL ;
tevent_req_received ( req ) ;
2011-05-05 20:12:07 +04:00
return status ;
}
2011-07-08 19:57:56 +04:00
talloc_steal ( mem_ctx , state - > recv_iov ) ;
2011-05-05 20:12:07 +04:00
* data_length = state - > data_length ;
* data = state - > data ;
2015-11-27 21:10:01 +03:00
tevent_req_received ( req ) ;
return status ;
2011-05-05 20:12:07 +04:00
}
2012-05-10 20:30:25 +04:00
NTSTATUS smb2cli_read ( struct smbXcli_conn * conn ,
uint32_t timeout_msec ,
struct smbXcli_session * session ,
2012-07-24 00:32:49 +04:00
struct smbXcli_tcon * tcon ,
2011-05-05 20:12:07 +04:00
uint32_t length ,
uint64_t offset ,
uint64_t fid_persistent ,
uint64_t fid_volatile ,
uint64_t minimum_count ,
uint64_t remaining_bytes ,
TALLOC_CTX * mem_ctx ,
uint8_t * * data ,
uint32_t * data_length )
{
TALLOC_CTX * frame = talloc_stackframe ( ) ;
2012-05-14 11:56:47 +04:00
struct tevent_context * ev ;
2011-05-05 20:12:07 +04:00
struct tevent_req * req ;
NTSTATUS status = NT_STATUS_NO_MEMORY ;
2012-05-10 20:30:25 +04:00
if ( smbXcli_conn_has_async_calls ( conn ) ) {
2011-05-05 20:12:07 +04:00
/*
* Can ' t use sync call while an async call is in flight
*/
status = NT_STATUS_INVALID_PARAMETER ;
goto fail ;
}
2013-02-18 12:07:11 +04:00
ev = samba_tevent_context_init ( frame ) ;
2011-05-05 20:12:07 +04:00
if ( ev = = NULL ) {
goto fail ;
}
2012-05-10 20:30:25 +04:00
req = smb2cli_read_send ( frame , ev ,
2012-07-24 00:32:49 +04:00
conn , timeout_msec , session , tcon ,
2012-05-10 20:30:25 +04:00
length , offset ,
2011-05-05 20:12:07 +04:00
fid_persistent , fid_volatile ,
minimum_count , remaining_bytes ) ;
if ( req = = NULL ) {
goto fail ;
}
if ( ! tevent_req_poll_ntstatus ( req , ev , & status ) ) {
goto fail ;
}
status = smb2cli_read_recv ( req , mem_ctx , data , data_length ) ;
fail :
TALLOC_FREE ( frame ) ;
return status ;
}