2005-11-11 12:37:16 +00:00
/*
Unix SMB / CIFS implementation .
SMB2 client tree handling
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-11 12:37:16 +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-11 12:37:16 +00:00
*/
# include "includes.h"
# include "libcli/raw/libcliraw.h"
2008-04-02 04:53:27 +02:00
# include "libcli/raw/raw_proto.h"
2005-11-11 12:37:16 +00:00
# include "libcli/smb2/smb2.h"
# include "libcli/smb2/smb2_calls.h"
2008-05-28 15:27:50 +10:00
# include "librpc/gen_ndr/ndr_security.h"
/*
parse a set of SMB2 create blobs
*/
NTSTATUS smb2_create_blob_parse ( TALLOC_CTX * mem_ctx , const DATA_BLOB buffer ,
struct smb2_create_blobs * blobs )
{
const uint8_t * data = buffer . data ;
uint32_t remaining = buffer . length ;
while ( remaining > 0 ) {
uint32_t next ;
uint32_t name_offset , name_length ;
uint32_t reserved , data_offset ;
uint32_t data_length ;
char * tag ;
DATA_BLOB b ;
NTSTATUS status ;
if ( remaining < 16 ) {
return NT_STATUS_INVALID_PARAMETER ;
}
next = IVAL ( data , 0 ) ;
name_offset = SVAL ( data , 4 ) ;
name_length = SVAL ( data , 6 ) ;
reserved = SVAL ( data , 8 ) ;
data_offset = SVAL ( data , 10 ) ;
data_length = IVAL ( data , 12 ) ;
if ( ( next & 0x7 ) ! = 0 | |
next > remaining | |
name_offset < 16 | |
name_offset > remaining | |
2008-05-28 16:58:34 +10:00
name_length ! = 4 | | /* windows enforces this */
2008-05-28 15:27:50 +10:00
name_offset + name_length > remaining | |
data_offset < name_offset + name_length | |
data_offset > remaining | |
data_offset + ( uint64_t ) data_length > remaining ) {
return NT_STATUS_INVALID_PARAMETER ;
}
tag = talloc_strndup ( mem_ctx , ( const char * ) data + name_offset , name_length ) ;
if ( tag = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
b = data_blob_const ( data + data_offset , data_length ) ;
status = smb2_create_blob_add ( mem_ctx , blobs , tag , b ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
talloc_free ( tag ) ;
if ( next = = 0 ) break ;
remaining - = next ;
data + = next ;
if ( remaining < 16 ) {
return NT_STATUS_INVALID_PARAMETER ;
}
}
return NT_STATUS_OK ;
}
2005-11-11 12:37:16 +00:00
2005-11-18 09:25:25 +00:00
/*
add a blob to a smb2_create attribute blob
*/
2008-04-16 10:03:08 +02:00
static NTSTATUS smb2_create_blob_push_one ( TALLOC_CTX * mem_ctx , DATA_BLOB * buffer ,
const struct smb2_create_blob * blob ,
bool last )
2005-11-18 09:25:25 +00:00
{
2008-04-16 10:03:08 +02:00
uint32_t ofs = buffer - > length ;
size_t tag_length = strlen ( blob - > tag ) ;
uint8_t pad = smb2_padding_size ( blob - > data . length + tag_length , 4 ) ;
if ( ! data_blob_realloc ( mem_ctx , buffer ,
buffer - > length + 0x14 + tag_length + blob - > data . length + pad ) )
2007-08-29 13:07:03 +00:00
return NT_STATUS_NO_MEMORY ;
2008-04-16 10:03:08 +02:00
2005-11-18 09:25:25 +00:00
if ( last ) {
2008-04-16 10:03:08 +02:00
SIVAL ( buffer - > data , ofs + 0x00 , 0 ) ;
2005-11-18 09:25:25 +00:00
} else {
2008-04-16 10:03:08 +02:00
SIVAL ( buffer - > data , ofs + 0x00 , 0x14 + tag_length + blob - > data . length + pad ) ;
2005-11-18 09:25:25 +00:00
}
2008-04-16 10:03:08 +02:00
SSVAL ( buffer - > data , ofs + 0x04 , 0x10 ) ; /* offset of tag */
SIVAL ( buffer - > data , ofs + 0x06 , tag_length ) ; /* tag length */
SSVAL ( buffer - > data , ofs + 0x0A , 0x14 + tag_length ) ; /* offset of data */
SIVAL ( buffer - > data , ofs + 0x0C , blob - > data . length ) ;
memcpy ( buffer - > data + ofs + 0x10 , blob - > tag , tag_length ) ;
SIVAL ( buffer - > data , ofs + 0x10 + tag_length , 0 ) ; /* pad? */
memcpy ( buffer - > data + ofs + 0x14 + tag_length , blob - > data . data , blob - > data . length ) ;
memset ( buffer - > data + ofs + 0x14 + tag_length + blob - > data . length , 0 , pad ) ;
return NT_STATUS_OK ;
}
2008-05-28 16:27:38 +10:00
/*
create a buffer of a set of create blobs
*/
NTSTATUS smb2_create_blob_push ( TALLOC_CTX * mem_ctx , DATA_BLOB * buffer ,
const struct smb2_create_blobs blobs )
{
int i ;
NTSTATUS status ;
* buffer = data_blob ( NULL , 0 ) ;
for ( i = 0 ; i < blobs . num_blobs ; i + + ) {
bool last = false ;
const struct smb2_create_blob * c ;
if ( ( i + 1 ) = = blobs . num_blobs ) {
last = true ;
}
c = & blobs . blobs [ i ] ;
status = smb2_create_blob_push_one ( mem_ctx , buffer , c , last ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
}
return NT_STATUS_OK ;
}
2008-04-16 10:03:08 +02:00
NTSTATUS smb2_create_blob_add ( TALLOC_CTX * mem_ctx , struct smb2_create_blobs * b ,
const char * tag , DATA_BLOB data )
{
struct smb2_create_blob * array ;
array = talloc_realloc ( mem_ctx , b - > blobs ,
struct smb2_create_blob ,
b - > num_blobs + 1 ) ;
NT_STATUS_HAVE_NO_MEMORY ( array ) ;
b - > blobs = array ;
b - > blobs [ b - > num_blobs ] . tag = talloc_strdup ( b - > blobs , tag ) ;
NT_STATUS_HAVE_NO_MEMORY ( b - > blobs [ b - > num_blobs ] . tag ) ;
if ( data . data ) {
b - > blobs [ b - > num_blobs ] . data = data_blob_talloc ( b - > blobs ,
data . data ,
data . length ) ;
NT_STATUS_HAVE_NO_MEMORY ( b - > blobs [ b - > num_blobs ] . data . data ) ;
} else {
b - > blobs [ b - > num_blobs ] . data = data_blob ( NULL , 0 ) ;
}
b - > num_blobs + = 1 ;
2005-11-18 09:25:25 +00:00
return NT_STATUS_OK ;
}
2005-11-11 12:37:16 +00:00
/*
send a create request
*/
struct smb2_request * smb2_create_send ( struct smb2_tree * tree , struct smb2_create * io )
{
struct smb2_request * req ;
NTSTATUS status ;
2008-05-28 16:27:38 +10:00
DATA_BLOB blob ;
2008-05-28 16:58:34 +10:00
struct smb2_create_blobs blobs ;
int i ;
ZERO_STRUCT ( blobs ) ;
2005-11-11 12:37:16 +00:00
2007-10-06 22:28:14 +00:00
req = smb2_request_init_tree ( tree , SMB2_OP_CREATE , 0x38 , true , 0 ) ;
2005-11-11 12:37:16 +00:00
if ( req = = NULL ) return NULL ;
2008-02-13 15:05:44 +11:00
SCVAL ( req - > out . body , 0x02 , io - > in . security_flags ) ;
SCVAL ( req - > out . body , 0x03 , io - > in . oplock_level ) ;
SIVAL ( req - > out . body , 0x04 , io - > in . impersonation_level ) ;
SBVAL ( req - > out . body , 0x08 , io - > in . create_flags ) ;
SBVAL ( req - > out . body , 0x10 , io - > in . reserved ) ;
SIVAL ( req - > out . body , 0x18 , io - > in . desired_access ) ;
SIVAL ( req - > out . body , 0x1C , io - > in . file_attributes ) ;
2005-11-11 14:04:46 +00:00
SIVAL ( req - > out . body , 0x20 , io - > in . share_access ) ;
2008-02-13 15:05:44 +11:00
SIVAL ( req - > out . body , 0x24 , io - > in . create_disposition ) ;
2005-11-11 14:04:46 +00:00
SIVAL ( req - > out . body , 0x28 , io - > in . create_options ) ;
2005-11-11 12:37:16 +00:00
2005-11-17 03:32:38 +00:00
status = smb2_push_o16s16_string ( & req - > out , 0x2C , io - > in . fname ) ;
2005-11-16 11:01:15 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( req ) ;
return NULL ;
}
2005-11-11 12:37:16 +00:00
2008-05-28 15:27:50 +10:00
/* now add all the optional blobs */
2005-11-18 09:25:25 +00:00
if ( io - > in . eas . num_eas ! = 0 ) {
DATA_BLOB b = data_blob_talloc ( req , NULL ,
2008-05-23 00:07:12 +10:00
ea_list_size_chained ( io - > in . eas . num_eas , io - > in . eas . eas , 4 ) ) ;
ea_put_list_chained ( b . data , io - > in . eas . num_eas , io - > in . eas . eas , 4 ) ;
2008-05-27 12:42:19 +10:00
status = smb2_create_blob_add ( req , & blobs ,
2008-04-16 10:03:08 +02:00
SMB2_CREATE_TAG_EXTA , b ) ;
2005-11-18 09:25:25 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( req ) ;
return NULL ;
}
data_blob_free ( & b ) ;
}
2005-11-18 11:45:24 +00:00
/* an empty MxAc tag seems to be used to ask the server to
return the maximum access mask allowed on the file */
2008-05-28 15:27:50 +10:00
if ( io - > in . query_maximal_access ) {
/* TODO: MS-SMB2 2.2.13.2.5 says this can contain a timestamp? What to do
with that if it doesn ' t match ? */
status = smb2_create_blob_add ( req , & blobs ,
SMB2_CREATE_TAG_MXAC , data_blob ( NULL , 0 ) ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( req ) ;
return NULL ;
}
}
if ( io - > in . alloc_size ! = 0 ) {
uint8_t data [ 8 ] ;
SBVAL ( data , 0 , io - > in . alloc_size ) ;
status = smb2_create_blob_add ( req , & blobs ,
SMB2_CREATE_TAG_ALSI , data_blob_const ( data , 8 ) ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( req ) ;
return NULL ;
}
}
if ( io - > in . durable_open ) {
status = smb2_create_blob_add ( req , & blobs ,
SMB2_CREATE_TAG_DHNQ , data_blob_talloc_zero ( req , 16 ) ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( req ) ;
return NULL ;
}
}
if ( io - > in . durable_handle ) {
uint8_t data [ 16 ] ;
smb2_push_handle ( data , io - > in . durable_handle ) ;
status = smb2_create_blob_add ( req , & blobs ,
SMB2_CREATE_TAG_DHNC , data_blob_const ( data , 16 ) ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( req ) ;
return NULL ;
}
}
if ( io - > in . timewarp ) {
uint8_t data [ 8 ] ;
SBVAL ( data , 0 , io - > in . timewarp ) ;
status = smb2_create_blob_add ( req , & blobs ,
SMB2_CREATE_TAG_TWRP , data_blob_const ( data , 8 ) ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( req ) ;
return NULL ;
}
2005-11-18 09:25:25 +00:00
}
2008-04-16 10:03:08 +02:00
2008-05-28 15:27:50 +10:00
if ( io - > in . sec_desc ) {
enum ndr_err_code ndr_err ;
DATA_BLOB sd_blob ;
ndr_err = ndr_push_struct_blob ( & sd_blob , req , NULL ,
io - > in . sec_desc ,
( ndr_push_flags_fn_t ) ndr_push_security_descriptor ) ;
if ( ! NDR_ERR_CODE_IS_SUCCESS ( ndr_err ) ) {
talloc_free ( req ) ;
return NULL ;
}
status = smb2_create_blob_add ( req , & blobs ,
SMB2_CREATE_TAG_SECD , sd_blob ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( req ) ;
return NULL ;
}
}
if ( io - > in . query_on_disk_id ) {
status = smb2_create_blob_add ( req , & blobs ,
SMB2_CREATE_TAG_QFID , data_blob ( NULL , 0 ) ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( req ) ;
return NULL ;
}
}
/* and any custom blobs */
2008-05-28 16:58:34 +10:00
for ( i = 0 ; i < io - > in . blobs . num_blobs ; i + + ) {
status = smb2_create_blob_add ( req , & blobs ,
io - > in . blobs . blobs [ i ] . tag ,
io - > in . blobs . blobs [ i ] . data ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( req ) ;
return NULL ;
}
}
2008-05-28 16:27:38 +10:00
status = smb2_create_blob_push ( req , & blob , blobs ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( req ) ;
return NULL ;
2008-04-16 10:03:08 +02:00
}
2005-11-18 09:25:25 +00:00
status = smb2_push_o32s32_blob ( & req - > out , 0x30 , blob ) ;
2005-11-16 11:01:15 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( req ) ;
return NULL ;
}
2005-11-11 12:37:16 +00:00
2008-05-28 16:27:38 +10:00
data_blob_free ( & blob ) ;
2005-11-11 12:37:16 +00:00
smb2_transport_send ( req ) ;
return req ;
}
/*
recv a create reply
*/
2005-11-16 11:01:15 +00:00
NTSTATUS smb2_create_recv ( struct smb2_request * req , TALLOC_CTX * mem_ctx , struct smb2_create * io )
2005-11-11 12:37:16 +00:00
{
2005-11-16 11:01:15 +00:00
NTSTATUS status ;
2008-05-28 15:27:50 +10:00
DATA_BLOB blob ;
int i ;
2005-11-16 11:01:15 +00:00
2005-11-11 12:37:16 +00:00
if ( ! smb2_request_receive ( req ) | |
2005-11-18 09:25:25 +00:00
! smb2_request_is_ok ( req ) ) {
2005-11-11 12:37:16 +00:00
return smb2_request_destroy ( req ) ;
}
2007-10-06 22:28:14 +00:00
SMB2_CHECK_PACKET_RECV ( req , 0x58 , true ) ;
2008-05-28 15:27:50 +10:00
ZERO_STRUCT ( io - > out ) ;
2008-02-13 15:05:44 +11:00
io - > out . oplock_level = CVAL ( req - > in . body , 0x02 ) ;
io - > out . reserved = CVAL ( req - > in . body , 0x03 ) ;
2005-11-11 23:27:47 +00:00
io - > out . create_action = IVAL ( req - > in . body , 0x04 ) ;
io - > out . create_time = smbcli_pull_nttime ( req - > in . body , 0x08 ) ;
io - > out . access_time = smbcli_pull_nttime ( req - > in . body , 0x10 ) ;
io - > out . write_time = smbcli_pull_nttime ( req - > in . body , 0x18 ) ;
io - > out . change_time = smbcli_pull_nttime ( req - > in . body , 0x20 ) ;
io - > out . alloc_size = BVAL ( req - > in . body , 0x28 ) ;
io - > out . size = BVAL ( req - > in . body , 0x30 ) ;
io - > out . file_attr = IVAL ( req - > in . body , 0x38 ) ;
2008-02-13 15:05:44 +11:00
io - > out . reserved2 = IVAL ( req - > in . body , 0x3C ) ;
2006-05-20 10:46:38 +00:00
smb2_pull_handle ( req - > in . body + 0x40 , & io - > out . file . handle ) ;
2008-05-28 15:27:50 +10:00
status = smb2_pull_o32s32_blob ( & req - > in , mem_ctx , req - > in . body + 0x50 , & blob ) ;
2005-11-16 11:01:15 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
smb2_request_destroy ( req ) ;
return status ;
}
2005-11-11 12:37:16 +00:00
2008-05-28 15:27:50 +10:00
status = smb2_create_blob_parse ( mem_ctx , blob , & io - > out . blobs ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
smb2_request_destroy ( req ) ;
return status ;
}
/* pull out the parsed blobs */
for ( i = 0 ; i < io - > out . blobs . num_blobs ; i + + ) {
if ( strcmp ( io - > out . blobs . blobs [ i ] . tag , SMB2_CREATE_TAG_MXAC ) = = 0 ) {
2008-05-28 21:48:26 +10:00
/* TODO: this also contains a status field in
first 4 bytes */
2008-05-28 15:27:50 +10:00
if ( io - > out . blobs . blobs [ i ] . data . length ! = 8 ) {
smb2_request_destroy ( req ) ;
return NT_STATUS_INVALID_NETWORK_RESPONSE ;
}
io - > out . maximal_access = IVAL ( io - > out . blobs . blobs [ i ] . data . data , 0 ) ;
}
if ( strcmp ( io - > out . blobs . blobs [ i ] . tag , SMB2_CREATE_TAG_QFID ) = = 0 ) {
if ( io - > out . blobs . blobs [ i ] . data . length ! = 32 ) {
smb2_request_destroy ( req ) ;
return NT_STATUS_INVALID_NETWORK_RESPONSE ;
}
memcpy ( io - > out . on_disk_id , io - > out . blobs . blobs [ i ] . data . data , 32 ) ;
}
}
data_blob_free ( & blob ) ;
2005-11-11 12:37:16 +00:00
return smb2_request_destroy ( req ) ;
}
/*
sync create request
*/
2005-11-16 11:01:15 +00:00
NTSTATUS smb2_create ( struct smb2_tree * tree , TALLOC_CTX * mem_ctx , struct smb2_create * io )
2005-11-11 12:37:16 +00:00
{
struct smb2_request * req = smb2_create_send ( tree , io ) ;
2005-11-16 11:01:15 +00:00
return smb2_create_recv ( req , mem_ctx , io ) ;
2005-11-11 12:37:16 +00:00
}