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"
# include "client.h"
# include "async_smb.h"
2011-09-07 20:44:21 +04:00
# include "../libcli/smb/smbXcli_base.h"
2011-05-05 20:12:07 +04:00
# include "smb2cli.h"
# include "libsmb/proto.h"
# include "lib/util/tevent_ntstatus.h"
struct smb2cli_tcon_state {
struct cli_state * cli ;
uint8_t fixed [ 8 ] ;
2011-09-05 20:22:57 +04:00
uint8_t dyn_pad [ 1 ] ;
2011-05-05 20:12:07 +04:00
} ;
static void smb2cli_tcon_done ( struct tevent_req * subreq ) ;
struct tevent_req * smb2cli_tcon_send ( TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
struct cli_state * cli ,
const char * share )
{
struct tevent_req * req , * subreq ;
struct smb2cli_tcon_state * state ;
uint8_t * fixed ;
const char * tcon_share ;
uint8_t * dyn ;
size_t dyn_len ;
req = tevent_req_create ( mem_ctx , & state , struct smb2cli_tcon_state ) ;
if ( req = = NULL ) {
return NULL ;
}
state - > cli = cli ;
2012-05-03 14:05:13 +04:00
tcon_share = talloc_asprintf ( state , " \\ \\ %s \\ %s " ,
2012-09-28 21:11:14 +04:00
smbXcli_conn_remote_name ( cli - > conn ) ,
share ) ;
2011-05-05 20:12:07 +04:00
if ( tevent_req_nomem ( tcon_share , req ) ) {
return tevent_req_post ( req , ev ) ;
}
if ( ! convert_string_talloc ( state , CH_UNIX , CH_UTF16 ,
2011-09-05 20:22:57 +04:00
tcon_share , strlen ( tcon_share ) ,
2011-05-05 20:12:07 +04:00
& dyn , & dyn_len ) ) {
2011-07-10 19:09:11 +04:00
tevent_req_oom ( req ) ;
2011-05-05 20:12:07 +04:00
return tevent_req_post ( req , ev ) ;
}
2011-09-05 20:22:57 +04:00
if ( strlen ( tcon_share ) = = 0 ) {
TALLOC_FREE ( dyn ) ;
dyn_len = 0 ;
}
2011-05-05 20:12:07 +04:00
fixed = state - > fixed ;
SSVAL ( fixed , 0 , 9 ) ;
SSVAL ( fixed , 4 , SMB2_HDR_BODY + 8 ) ;
SSVAL ( fixed , 6 , dyn_len ) ;
2011-09-05 20:22:57 +04:00
if ( dyn_len = = 0 ) {
dyn = state - > dyn_pad ; ;
dyn_len = sizeof ( state - > dyn_pad ) ;
}
2011-09-07 20:44:21 +04:00
subreq = smb2cli_req_send ( state , ev , cli - > conn , SMB2_OP_TCON ,
2011-08-12 19:26:13 +04:00
0 , 0 , /* flags */
2011-09-17 21:56:50 +04:00
cli - > timeout ,
2012-07-25 12:36:27 +04:00
NULL , /* tcon */
2011-09-03 12:18:17 +04:00
cli - > smb2 . session ,
2011-05-05 20:12:07 +04:00
state - > fixed , sizeof ( state - > fixed ) ,
2013-08-13 12:25:52 +04:00
dyn , dyn_len ,
0 ) ; /* 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_tcon_done , req ) ;
return req ;
}
static void smb2cli_tcon_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct smb2cli_tcon_state * state = tevent_req_data (
req , struct smb2cli_tcon_state ) ;
struct cli_state * cli = state - > cli ;
NTSTATUS status ;
struct iovec * iov ;
uint8_t * body ;
2012-07-25 11:01:02 +04:00
uint32_t tcon_id ;
2012-07-25 11:05:32 +04:00
uint8_t share_type ;
uint32_t share_flags ;
uint32_t share_capabilities ;
uint32_t maximal_access ;
2011-08-31 02:40:06 +04:00
static const struct smb2cli_req_expected_response expected [ ] = {
{
. status = NT_STATUS_OK ,
. body_size = 0x10
}
} ;
2011-05-05 20:12:07 +04:00
2012-05-03 14:05:13 +04:00
status = smb2cli_req_recv ( subreq , state , & iov ,
2011-08-31 02:40:06 +04:00
expected , ARRAY_SIZE ( expected ) ) ;
2012-07-25 11:05:32 +04:00
TALLOC_FREE ( subreq ) ;
2011-05-05 20:12:07 +04:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
tevent_req_nterror ( req , status ) ;
return ;
}
2012-07-25 11:01:02 +04:00
tcon_id = IVAL ( iov [ 0 ] . iov_base , SMB2_HDR_TID ) ;
2011-05-05 20:12:07 +04:00
body = ( uint8_t * ) iov [ 1 ] . iov_base ;
2012-07-25 11:05:32 +04:00
share_type = CVAL ( body , 0x02 ) ;
share_flags = IVAL ( body , 0x04 ) ;
share_capabilities = IVAL ( body , 0x08 ) ;
maximal_access = IVAL ( body , 0x0C ) ;
2012-07-24 00:22:02 +04:00
cli - > smb2 . tcon = smbXcli_tcon_create ( cli ) ;
if ( tevent_req_nomem ( cli - > smb2 . tcon , req ) ) {
return ;
}
smb2cli_tcon_set_values ( cli - > smb2 . tcon ,
2012-07-25 14:33:39 +04:00
cli - > smb2 . session ,
2012-07-25 11:01:02 +04:00
tcon_id ,
2012-07-25 11:05:32 +04:00
share_type ,
share_flags ,
share_capabilities ,
maximal_access ) ;
2012-07-24 00:22:02 +04:00
2011-05-05 20:12:07 +04:00
tevent_req_done ( req ) ;
}
NTSTATUS smb2cli_tcon_recv ( struct tevent_req * req )
{
return tevent_req_simple_recv_ntstatus ( req ) ;
}
NTSTATUS smb2cli_tcon ( struct cli_state * cli , const char * share )
{
TALLOC_CTX * frame = talloc_stackframe ( ) ;
2012-05-14 11:55:42 +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-26 13:45:09 +04:00
if ( smbXcli_conn_has_async_calls ( cli - > 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:08:19 +04:00
ev = samba_tevent_context_init ( frame ) ;
2011-05-05 20:12:07 +04:00
if ( ev = = NULL ) {
goto fail ;
}
req = smb2cli_tcon_send ( frame , ev , cli , share ) ;
if ( req = = NULL ) {
goto fail ;
}
if ( ! tevent_req_poll_ntstatus ( req , ev , & status ) ) {
goto fail ;
}
status = smb2cli_tcon_recv ( req ) ;
fail :
TALLOC_FREE ( frame ) ;
return status ;
}
struct smb2cli_tdis_state {
2011-11-21 19:30:09 +04:00
struct cli_state * cli ;
2011-05-05 20:12:07 +04:00
uint8_t fixed [ 4 ] ;
} ;
static void smb2cli_tdis_done ( struct tevent_req * subreq ) ;
struct tevent_req * smb2cli_tdis_send ( TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
struct cli_state * cli )
{
struct tevent_req * req , * subreq ;
struct smb2cli_tdis_state * state ;
req = tevent_req_create ( mem_ctx , & state ,
struct smb2cli_tdis_state ) ;
if ( req = = NULL ) {
return NULL ;
}
2011-11-21 19:30:09 +04:00
state - > cli = cli ;
2011-05-05 20:12:07 +04:00
SSVAL ( state - > fixed , 0 , 4 ) ;
2011-09-07 20:44:21 +04:00
subreq = smb2cli_req_send ( state , ev , cli - > conn , SMB2_OP_TDIS ,
2011-08-12 19:26:13 +04:00
0 , 0 , /* flags */
2011-09-17 21:56:50 +04:00
cli - > timeout ,
2012-07-25 12:36:27 +04:00
cli - > smb2 . tcon ,
2011-09-03 12:18:17 +04:00
cli - > smb2 . session ,
2011-05-05 20:12:07 +04:00
state - > fixed , sizeof ( state - > fixed ) ,
2013-08-13 12:25:52 +04:00
NULL , 0 , /* dyn* */
0 ) ; /* 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_tdis_done , req ) ;
return req ;
}
static void smb2cli_tdis_done ( struct tevent_req * subreq )
{
struct tevent_req * req =
tevent_req_callback_data ( subreq ,
struct tevent_req ) ;
2011-11-21 19:30:09 +04:00
struct smb2cli_tdis_state * state =
tevent_req_data ( req ,
struct smb2cli_tdis_state ) ;
2011-05-05 20:12:07 +04:00
NTSTATUS status ;
2011-08-31 02:40:06 +04:00
static const struct smb2cli_req_expected_response expected [ ] = {
{
. status = NT_STATUS_OK ,
. body_size = 0x04
}
} ;
2011-05-05 20:12:07 +04:00
2012-05-03 14:04:12 +04:00
status = smb2cli_req_recv ( subreq , NULL , NULL ,
2011-08-31 02:40:06 +04:00
expected , ARRAY_SIZE ( expected ) ) ;
2011-05-05 20:12:07 +04:00
TALLOC_FREE ( subreq ) ;
if ( tevent_req_nterror ( req , status ) ) {
return ;
}
2013-08-14 13:44:10 +04:00
smb2cli_tcon_set_values ( state - > cli - > smb2 . tcon , NULL ,
UINT32_MAX , 0 , 0 , 0 , 0 ) ;
2011-05-05 20:12:07 +04:00
tevent_req_done ( req ) ;
}
NTSTATUS smb2cli_tdis_recv ( struct tevent_req * req )
{
return tevent_req_simple_recv_ntstatus ( req ) ;
}
NTSTATUS smb2cli_tdis ( struct cli_state * cli )
{
TALLOC_CTX * frame = talloc_stackframe ( ) ;
2012-05-14 11:55:42 +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-26 13:45:09 +04:00
if ( smbXcli_conn_has_async_calls ( cli - > 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:08:19 +04:00
ev = samba_tevent_context_init ( frame ) ;
2011-05-05 20:12:07 +04:00
if ( ev = = NULL ) {
goto fail ;
}
req = smb2cli_tdis_send ( frame , ev , cli ) ;
if ( req = = NULL ) {
goto fail ;
}
if ( ! tevent_req_poll_ntstatus ( req , ev , & status ) ) {
goto fail ;
}
status = smb2cli_tdis_recv ( req ) ;
fail :
TALLOC_FREE ( frame ) ;
return status ;
}