2005-11-25 05:23:55 +00:00
/*
Unix SMB / CIFS implementation .
2005-12-01 00:18:29 +00:00
SMB2 client ioctl call
2005-11-25 05:23:55 +00:00
Copyright ( C ) Andrew Tridgell 2005
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-10 02:07:03 +00:00
the Free Software Foundation ; either version 3 of the License , or
2005-11-25 05:23:55 +00:00
( 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
2007-07-10 02:07:03 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2005-11-25 05:23:55 +00:00
*/
# include "includes.h"
# include "libcli/smb2/smb2.h"
# include "libcli/smb2/smb2_calls.h"
2013-01-15 17:23:01 +01:00
# include "librpc/gen_ndr/ioctl.h"
2005-11-25 05:23:55 +00:00
/*
2005-12-01 00:18:29 +00:00
send a ioctl request
2005-11-25 05:23:55 +00:00
*/
2005-12-01 00:18:29 +00:00
struct smb2_request * smb2_ioctl_send ( struct smb2_tree * tree , struct smb2_ioctl * io )
2005-11-25 05:23:55 +00:00
{
NTSTATUS status ;
struct smb2_request * req ;
2019-01-08 16:13:41 +01:00
uint64_t max_payload_in ;
uint64_t max_payload_out ;
size_t max_payload ;
2005-11-25 05:23:55 +00:00
2007-10-06 22:28:14 +00:00
req = smb2_request_init_tree ( tree , SMB2_OP_IOCTL , 0x38 , true ,
2005-11-25 05:23:55 +00:00
io - > in . in . length + io - > in . out . length ) ;
if ( req = = NULL ) return NULL ;
2005-12-02 03:17:40 +00:00
SSVAL ( req - > out . body , 0x02 , 0 ) ; /* pad */
2005-12-01 00:18:29 +00:00
SIVAL ( req - > out . body , 0x04 , io - > in . function ) ;
2006-05-20 10:46:38 +00:00
smb2_push_handle ( req - > out . body + 0x08 , & io - > in . file . handle ) ;
2005-11-25 05:23:55 +00:00
status = smb2_push_o32s32_blob ( & req - > out , 0x18 , io - > in . out ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( req ) ;
return NULL ;
}
2019-01-08 15:52:35 +01:00
SIVAL ( req - > out . body , 0x20 , io - > in . max_input_response ) ;
2005-11-25 08:24:36 +00:00
2005-11-25 05:23:55 +00:00
status = smb2_push_o32s32_blob ( & req - > out , 0x24 , io - > in . in ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( req ) ;
return NULL ;
}
2019-01-08 15:52:35 +01:00
SIVAL ( req - > out . body , 0x2C , io - > in . max_output_response ) ;
2005-11-25 08:24:36 +00:00
SBVAL ( req - > out . body , 0x30 , io - > in . flags ) ;
2019-01-08 16:13:41 +01:00
max_payload_in = io - > in . out . length + io - > in . in . length ;
max_payload_in = MIN ( max_payload_in , UINT32_MAX ) ;
max_payload_out = io - > in . max_input_response + io - > in . max_output_response ;
max_payload_out = MIN ( max_payload_out , UINT32_MAX ) ;
max_payload = MAX ( max_payload_in , max_payload_out ) ;
req - > credit_charge = ( MAX ( max_payload , 1 ) - 1 ) / 65536 + 1 ;
2005-11-25 05:23:55 +00:00
smb2_transport_send ( req ) ;
return req ;
}
2013-01-15 17:23:01 +01:00
/*
* 3.3 .4 .4 Sending an Error Response
*/
static bool smb2_ioctl_is_failure ( uint32_t ctl_code , NTSTATUS status ,
size_t data_size )
{
if ( NT_STATUS_IS_OK ( status ) ) {
return false ;
}
if ( NT_STATUS_EQUAL ( status , STATUS_BUFFER_OVERFLOW )
& & ( ( ctl_code = = FSCTL_PIPE_TRANSCEIVE )
| | ( ctl_code = = FSCTL_PIPE_PEEK )
| | ( ctl_code = = FSCTL_DFS_GET_REFERRALS ) ) ) {
return false ;
}
if ( ( ( ctl_code = = FSCTL_SRV_COPYCHUNK )
| | ( ctl_code = = FSCTL_SRV_COPYCHUNK_WRITE ) )
& & ( data_size = = sizeof ( struct srv_copychunk_rsp ) ) ) {
/*
* copychunk responses may come with copychunk data or error
* response data , independent of status .
*/
return false ;
}
return true ;
}
2005-11-25 05:23:55 +00:00
/*
2005-12-01 00:18:29 +00:00
recv a ioctl reply
2005-11-25 05:23:55 +00:00
*/
2013-01-15 17:23:01 +01:00
NTSTATUS smb2_ioctl_recv ( struct smb2_request * req ,
2005-12-01 00:18:29 +00:00
TALLOC_CTX * mem_ctx , struct smb2_ioctl * io )
2005-11-25 05:23:55 +00:00
{
NTSTATUS status ;
2013-01-15 17:23:01 +01:00
if ( ! smb2_request_receive ( req ) | |
smb2_ioctl_is_failure ( io - > in . function , req - > status ,
req - > in . bufinfo . data_size ) ) {
2005-11-25 05:23:55 +00:00
return smb2_request_destroy ( req ) ;
}
2007-10-06 22:28:14 +00:00
SMB2_CHECK_PACKET_RECV ( req , 0x30 , true ) ;
2005-11-25 05:23:55 +00:00
2019-01-08 15:52:35 +01:00
io - > out . reserved = SVAL ( req - > in . body , 0x02 ) ;
2005-12-01 00:18:29 +00:00
io - > out . function = IVAL ( req - > in . body , 0x04 ) ;
2006-05-20 10:46:38 +00:00
smb2_pull_handle ( req - > in . body + 0x08 , & io - > out . file . handle ) ;
2005-11-25 08:24:36 +00:00
2005-11-25 05:23:55 +00:00
status = smb2_pull_o32s32_blob ( & req - > in , mem_ctx , req - > in . body + 0x18 , & io - > out . in ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
smb2_request_destroy ( req ) ;
return status ;
}
status = smb2_pull_o32s32_blob ( & req - > in , mem_ctx , req - > in . body + 0x20 , & io - > out . out ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
smb2_request_destroy ( req ) ;
return status ;
}
2019-01-08 15:52:35 +01:00
io - > out . flags = IVAL ( req - > in . body , 0x28 ) ;
io - > out . reserved2 = IVAL ( req - > in . body , 0x2C ) ;
2005-11-25 05:23:55 +00:00
return smb2_request_destroy ( req ) ;
}
/*
2005-12-01 00:18:29 +00:00
sync ioctl request
2005-11-25 05:23:55 +00:00
*/
2005-12-01 00:18:29 +00:00
NTSTATUS smb2_ioctl ( struct smb2_tree * tree , TALLOC_CTX * mem_ctx , struct smb2_ioctl * io )
2005-11-25 05:23:55 +00:00
{
2005-12-01 00:18:29 +00:00
struct smb2_request * req = smb2_ioctl_send ( tree , io ) ;
return smb2_ioctl_recv ( req , mem_ctx , io ) ;
2005-11-25 05:23:55 +00:00
}