2007-11-30 18:02:19 -08:00
/*
2002-01-30 06:08:46 +00:00
Unix SMB / CIFS implementation .
2000-04-25 14:04:06 +00:00
client transaction calls
Copyright ( C ) Andrew Tridgell 1994 - 1998
2007-11-30 18:02:19 -08:00
2000-04-25 14:04:06 +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
2000-04-25 14:04:06 +00:00
( at your option ) any later version .
2007-11-30 18:02:19 -08:00
2000-04-25 14:04:06 +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 .
2007-11-30 18:02:19 -08:00
2000-04-25 14:04:06 +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/>.
2000-04-25 14:04:06 +00:00
*/
# include "includes.h"
2011-05-06 11:47:43 +02:00
# include "libsmb/libsmb.h"
2011-04-28 17:38:09 +02:00
# include "../lib/util/tevent_ntstatus.h"
2010-08-26 09:58:09 +02:00
# include "async_smb.h"
2011-08-28 18:05:04 +02:00
# include "../libcli/smb/smbXcli_base.h"
2008-09-09 14:47:23 +02:00
struct cli_trans_state {
struct cli_state * cli ;
2011-08-28 18:05:04 +02:00
struct tevent_req * req ;
struct cli_trans_state * * ptr ;
2008-09-09 14:47:23 +02:00
} ;
2011-06-08 10:31:23 +02:00
static int cli_trans_state_destructor ( struct cli_trans_state * state )
{
2011-08-28 18:05:04 +02:00
talloc_set_destructor ( state - > ptr , NULL ) ;
talloc_free ( state - > ptr ) ;
2011-06-08 10:31:23 +02:00
return 0 ;
}
2011-08-28 18:05:04 +02:00
static int cli_trans_state_ptr_destructor ( struct cli_trans_state * * ptr )
2009-04-05 20:51:52 +02:00
{
2011-08-28 18:05:04 +02:00
struct cli_trans_state * state = * ptr ;
void * parent = talloc_parent ( state ) ;
2008-09-09 14:47:23 +02:00
2011-08-28 18:05:04 +02:00
talloc_set_destructor ( state , NULL ) ;
2011-06-08 00:44:34 +02:00
2011-08-28 18:05:04 +02:00
talloc_reparent ( state , parent , state - > req ) ;
talloc_free ( state ) ;
return 0 ;
2008-09-09 14:47:23 +02:00
}
2009-04-05 20:51:52 +02:00
struct tevent_req * cli_trans_send (
2011-10-26 15:43:08 +02:00
TALLOC_CTX * mem_ctx , struct tevent_context * ev ,
2009-04-05 20:51:52 +02:00
struct cli_state * cli , uint8_t cmd ,
const char * pipe_name , uint16_t fid , uint16_t function , int flags ,
uint16_t * setup , uint8_t num_setup , uint8_t max_setup ,
uint8_t * param , uint32_t num_param , uint32_t max_param ,
uint8_t * data , uint32_t num_data , uint32_t max_data )
2008-09-09 14:47:23 +02:00
{
2009-04-05 20:51:52 +02:00
struct cli_trans_state * state ;
2011-08-28 18:05:04 +02:00
uint8_t additional_flags = 0 ;
uint8_t clear_flags = 0 ;
uint16_t additional_flags2 = 0 ;
uint16_t clear_flags2 = 0 ;
2009-03-09 09:50:36 +01:00
2011-08-28 18:05:04 +02:00
state = talloc_zero ( mem_ctx , struct cli_trans_state ) ;
if ( state = = NULL ) {
2009-04-05 20:51:52 +02:00
return NULL ;
2008-09-09 14:47:23 +02:00
}
2009-04-05 20:51:52 +02:00
state - > cli = cli ;
2011-08-28 18:05:04 +02:00
state - > ptr = talloc ( state , struct cli_trans_state * ) ;
if ( state - > ptr = = NULL ) {
talloc_free ( state ) ;
return NULL ;
2008-09-09 14:47:23 +02:00
}
2011-08-28 18:05:04 +02:00
* state - > ptr = state ;
if ( cli - > case_sensitive ) {
clear_flags | = FLAG_CASELESS_PATHNAMES ;
} else {
/* Default setting, case insensitive. */
additional_flags | = FLAG_CASELESS_PATHNAMES ;
}
2012-05-19 18:23:40 +02:00
if ( ( smb1cli_conn_capabilities ( cli - > conn ) & CAP_DFS ) & & cli - > dfsroot ) {
2011-08-28 18:05:04 +02:00
additional_flags2 | = FLAGS2_DFS_PATHNAMES ;
}
state - > req = smb1cli_trans_send ( state , ev ,
cli - > conn , cmd ,
additional_flags , clear_flags ,
additional_flags2 , clear_flags2 ,
cli - > timeout ,
cli - > smb1 . pid , cli - > smb1 . tid , cli - > smb1 . uid ,
pipe_name , fid , function , flags ,
setup , num_setup , max_setup ,
param , num_param , max_param ,
data , num_data , max_data ) ;
if ( state - > req = = NULL ) {
talloc_free ( state ) ;
return NULL ;
2008-09-09 14:47:23 +02:00
}
2011-06-08 16:11:00 +02:00
2011-08-28 18:05:04 +02:00
talloc_reparent ( state , state - > req , state - > ptr ) ;
2011-06-08 10:31:23 +02:00
talloc_set_destructor ( state , cli_trans_state_destructor ) ;
2011-08-28 18:05:04 +02:00
talloc_set_destructor ( state - > ptr , cli_trans_state_ptr_destructor ) ;
2011-06-08 16:11:00 +02:00
2011-08-28 18:05:04 +02:00
return state - > req ;
2008-09-09 14:47:23 +02:00
}
2009-04-05 20:51:52 +02:00
NTSTATUS cli_trans_recv ( struct tevent_req * req , TALLOC_CTX * mem_ctx ,
2010-08-05 12:48:52 +02:00
uint16_t * recv_flags2 ,
2009-11-14 11:12:50 +01:00
uint16_t * * setup , uint8_t min_setup ,
uint8_t * num_setup ,
uint8_t * * param , uint32_t min_param ,
uint32_t * num_param ,
uint8_t * * data , uint32_t min_data ,
uint32_t * num_data )
2008-09-09 14:47:23 +02:00
{
2008-12-01 08:23:35 +01:00
NTSTATUS status ;
2011-08-28 18:05:04 +02:00
void * parent = talloc_parent ( req ) ;
struct cli_trans_state * state =
talloc_get_type ( parent ,
struct cli_trans_state ) ;
bool map_dos_errors = true ;
status = smb1cli_trans_recv ( req , mem_ctx , recv_flags2 ,
setup , min_setup , num_setup ,
param , min_param , num_param ,
data , min_data , num_data ) ;
if ( state ) {
map_dos_errors = state - > cli - > map_dos_errors ;
state - > cli - > raw_status = status ;
talloc_free ( state - > ptr ) ;
state = NULL ;
}
if ( NT_STATUS_IS_DOS ( status ) & & map_dos_errors ) {
uint8_t eclass = NT_STATUS_DOS_CLASS ( status ) ;
uint16_t ecode = NT_STATUS_DOS_CODE ( status ) ;
/*
* TODO : is it really a good idea to do a mapping here ?
*
* The old cli_pull_error ( ) also does it , so I do not change
* the behavior yet .
*/
status = dos_to_ntstatus ( eclass , ecode ) ;
2008-09-09 14:47:23 +02:00
}
2011-08-28 18:05:04 +02:00
return status ;
2008-09-09 14:47:23 +02:00
}
NTSTATUS cli_trans ( TALLOC_CTX * mem_ctx , struct cli_state * cli ,
uint8_t trans_cmd ,
const char * pipe_name , uint16_t fid , uint16_t function ,
int flags ,
uint16_t * setup , uint8_t num_setup , uint8_t max_setup ,
uint8_t * param , uint32_t num_param , uint32_t max_param ,
uint8_t * data , uint32_t num_data , uint32_t max_data ,
2010-08-05 12:48:52 +02:00
uint16_t * recv_flags2 ,
2009-11-14 11:12:50 +01:00
uint16_t * * rsetup , uint8_t min_rsetup , uint8_t * num_rsetup ,
uint8_t * * rparam , uint32_t min_rparam , uint32_t * num_rparam ,
uint8_t * * rdata , uint32_t min_rdata , uint32_t * num_rdata )
2008-09-09 14:47:23 +02:00
{
2011-08-28 18:05:04 +02:00
NTSTATUS status ;
uint8_t additional_flags = 0 ;
uint8_t clear_flags = 0 ;
uint16_t additional_flags2 = 0 ;
uint16_t clear_flags2 = 0 ;
2008-09-09 14:47:23 +02:00
2011-08-28 18:05:04 +02:00
if ( cli - > case_sensitive ) {
clear_flags | = FLAG_CASELESS_PATHNAMES ;
} else {
/* Default setting, case insensitive. */
additional_flags | = FLAG_CASELESS_PATHNAMES ;
}
2012-05-19 18:23:40 +02:00
if ( ( smb1cli_conn_capabilities ( cli - > conn ) & CAP_DFS ) & & cli - > dfsroot ) {
2011-08-28 18:05:04 +02:00
additional_flags2 | = FLAGS2_DFS_PATHNAMES ;
}
status = smb1cli_trans ( mem_ctx ,
cli - > conn , trans_cmd ,
additional_flags , clear_flags ,
additional_flags2 , clear_flags2 ,
cli - > timeout ,
cli - > smb1 . pid , cli - > smb1 . tid , cli - > smb1 . uid ,
pipe_name , fid , function , flags ,
setup , num_setup , max_setup ,
param , num_param , max_param ,
data , num_data , max_data ,
recv_flags2 ,
rsetup , min_rsetup , num_rsetup ,
rparam , min_rparam , num_rparam ,
rdata , min_rdata , num_rdata ) ;
cli - > raw_status = status ;
if ( NT_STATUS_IS_DOS ( status ) & & cli - > map_dos_errors ) {
uint8_t eclass = NT_STATUS_DOS_CLASS ( status ) ;
uint16_t ecode = NT_STATUS_DOS_CODE ( status ) ;
2008-09-09 14:47:23 +02:00
/*
2011-08-28 18:05:04 +02:00
* TODO : is it really a good idea to do a mapping here ?
*
* The old cli_pull_error ( ) also does it , so I do not change
* the behavior yet .
2008-09-09 14:47:23 +02:00
*/
2011-08-28 18:05:04 +02:00
status = dos_to_ntstatus ( eclass , ecode ) ;
2008-09-09 14:47:23 +02:00
}
return status ;
}