2003-08-13 05:53:07 +04:00
/*
Unix SMB / CIFS implementation .
transaction2 handling
Copyright ( C ) Andrew Tridgell 2003
2011-05-22 23:41:56 +04:00
Copyright Matthieu Patou < mat @ matws . net > 2010 - 2011
2003-08-13 05:53:07 +04: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-10 06:07:03 +04:00
the Free Software Foundation ; either version 3 of the License , or
2003-08-13 05:53:07 +04: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 06:07:03 +04:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2003-08-13 05:53:07 +04:00
*/
/*
This file handles the parsing of transact2 requests
*/
# include "includes.h"
2020-11-20 17:27:17 +03:00
# include "samba/service_stream.h"
2004-11-02 10:18:24 +03:00
# include "smb_server/smb_server.h"
2005-12-28 01:51:30 +03:00
# include "ntvfs/ntvfs.h"
2005-12-28 18:38:36 +03:00
# include "libcli/raw/libcliraw.h"
2008-04-02 06:53:27 +04:00
# include "libcli/raw/raw_proto.h"
2010-05-01 22:27:31 +04:00
# include "librpc/gen_ndr/dfsblobs.h"
# include "librpc/gen_ndr/ndr_dfsblobs.h"
# include "dsdb/samdb/samdb.h"
# include "auth/session.h"
# include "param/param.h"
# include "lib/tsocket/tsocket.h"
2011-10-04 14:42:11 +04:00
# include "dfs_server/dfs_server_ad.h"
2010-05-01 22:27:31 +04:00
# define MAX_DFS_RESPONSE 56*1024 /* 56 Kb */
2004-11-02 10:18:24 +03:00
2006-03-18 14:10:21 +03:00
# define TRANS2_CHECK_ASYNC_STATUS_SIMPLE do { \
if ( ! NT_STATUS_IS_OK ( req - > ntvfs - > async_states - > status ) ) { \
trans2_setup_reply ( trans , 0 , 0 , 0 ) ; \
return req - > ntvfs - > async_states - > status ; \
} \
} while ( 0 )
# define TRANS2_CHECK_ASYNC_STATUS(ptr, type) do { \
TRANS2_CHECK_ASYNC_STATUS_SIMPLE ; \
ptr = talloc_get_type ( op - > op_info , type ) ; \
} while ( 0 )
2006-06-19 22:01:27 +04:00
# define TRANS2_CHECK(cmd) do { \
NTSTATUS _status ; \
_status = cmd ; \
NT_STATUS_NOT_OK_RETURN ( _status ) ; \
} while ( 0 )
2006-03-18 14:10:21 +03:00
2006-03-17 12:42:04 +03:00
/*
hold the state of a nttrans op while in progress . Needed to allow for async backend
functions .
*/
struct trans_op {
struct smbsrv_request * req ;
struct smb_trans2 * trans ;
uint8_t command ;
NTSTATUS ( * send_fn ) ( struct trans_op * ) ;
void * op_info ;
} ;
2011-10-04 14:42:11 +04:00
2003-08-13 05:53:07 +04:00
# define CHECK_MIN_BLOB_SIZE(blob, size) do { \
if ( ( blob ) - > length < ( size ) ) { \
return NT_STATUS_INFO_LENGTH_MISMATCH ; \
} } while ( 0 )
/* setup a trans2 reply, given the data and params sizes */
2006-06-19 22:01:27 +04:00
static NTSTATUS trans2_setup_reply ( struct smb_trans2 * trans ,
uint16_t param_size , uint16_t data_size ,
2008-07-07 16:00:53 +04:00
uint8_t setup_count )
2003-08-13 05:53:07 +04:00
{
trans - > out . setup_count = setup_count ;
2006-06-19 22:01:27 +04:00
if ( setup_count > 0 ) {
2006-03-17 12:42:04 +03:00
trans - > out . setup = talloc_zero_array ( trans , uint16_t , setup_count ) ;
2006-06-19 22:01:27 +04:00
NT_STATUS_HAVE_NO_MEMORY ( trans - > out . setup ) ;
2003-08-13 05:53:07 +04:00
}
2006-03-17 12:42:04 +03:00
trans - > out . params = data_blob_talloc ( trans , NULL , param_size ) ;
2006-06-19 22:01:27 +04:00
if ( param_size > 0 ) NT_STATUS_HAVE_NO_MEMORY ( trans - > out . params . data ) ;
2006-03-17 12:42:04 +03:00
trans - > out . data = data_blob_talloc ( trans , NULL , data_size ) ;
2006-06-19 22:01:27 +04:00
if ( data_size > 0 ) NT_STATUS_HAVE_NO_MEMORY ( trans - > out . data . data ) ;
return NT_STATUS_OK ;
2003-08-13 05:53:07 +04:00
}
2006-06-21 23:22:40 +04:00
static NTSTATUS trans2_push_fsinfo ( struct smbsrv_connection * smb_conn ,
TALLOC_CTX * mem_ctx ,
DATA_BLOB * blob ,
union smb_fsinfo * fsinfo ,
int default_str_flags )
{
enum smb_fsinfo_level passthru_level ;
switch ( fsinfo - > generic . level ) {
case RAW_QFS_ALLOCATION :
2006-06-22 01:24:00 +04:00
TRANS2_CHECK ( smbsrv_blob_grow_data ( mem_ctx , blob , 18 ) ) ;
2006-06-21 23:22:40 +04:00
SIVAL ( blob - > data , 0 , fsinfo - > allocation . out . fs_id ) ;
SIVAL ( blob - > data , 4 , fsinfo - > allocation . out . sectors_per_unit ) ;
SIVAL ( blob - > data , 8 , fsinfo - > allocation . out . total_alloc_units ) ;
SIVAL ( blob - > data , 12 , fsinfo - > allocation . out . avail_alloc_units ) ;
SSVAL ( blob - > data , 16 , fsinfo - > allocation . out . bytes_per_sector ) ;
return NT_STATUS_OK ;
case RAW_QFS_VOLUME :
2006-06-22 01:24:00 +04:00
TRANS2_CHECK ( smbsrv_blob_grow_data ( mem_ctx , blob , 5 ) ) ;
2006-06-21 23:22:40 +04:00
SIVAL ( blob - > data , 0 , fsinfo - > volume . out . serial_number ) ;
/* w2k3 implements this incorrectly for unicode - it
* leaves the last byte off the string */
2006-06-22 01:24:00 +04:00
TRANS2_CHECK ( smbsrv_blob_append_string ( mem_ctx , blob ,
2006-06-21 23:22:40 +04:00
fsinfo - > volume . out . volume_name . s ,
4 , default_str_flags ,
STR_LEN8BIT | STR_NOALIGN ) ) ;
return NT_STATUS_OK ;
case RAW_QFS_VOLUME_INFO :
passthru_level = RAW_QFS_VOLUME_INFORMATION ;
break ;
case RAW_QFS_SIZE_INFO :
passthru_level = RAW_QFS_SIZE_INFORMATION ;
break ;
case RAW_QFS_DEVICE_INFO :
passthru_level = RAW_QFS_DEVICE_INFORMATION ;
break ;
case RAW_QFS_ATTRIBUTE_INFO :
passthru_level = RAW_QFS_ATTRIBUTE_INFORMATION ;
break ;
default :
passthru_level = fsinfo - > generic . level ;
break ;
}
2006-06-22 01:24:00 +04:00
return smbsrv_push_passthru_fsinfo ( mem_ctx , blob ,
2006-06-21 23:22:40 +04:00
passthru_level , fsinfo ,
default_str_flags ) ;
}
2006-06-20 08:14:37 +04:00
/*
trans2 qfsinfo implementation send
*/
static NTSTATUS trans2_qfsinfo_send ( struct trans_op * op )
{
struct smbsrv_request * req = op - > req ;
struct smb_trans2 * trans = op - > trans ;
union smb_fsinfo * fsinfo ;
TRANS2_CHECK_ASYNC_STATUS ( fsinfo , union smb_fsinfo ) ;
TRANS2_CHECK ( trans2_setup_reply ( trans , 0 , 0 , 0 ) ) ;
2006-06-20 10:32:29 +04:00
TRANS2_CHECK ( trans2_push_fsinfo ( req - > smb_conn , trans ,
& trans - > out . data , fsinfo ,
2006-06-22 01:24:00 +04:00
SMBSRV_REQ_DEFAULT_STR_FLAGS ( req ) ) ) ;
2006-06-20 08:14:37 +04:00
return NT_STATUS_OK ;
}
2006-03-17 12:42:04 +03:00
/*
trans2 qfsinfo implementation
*/
static NTSTATUS trans2_qfsinfo ( struct smbsrv_request * req , struct trans_op * op )
{
struct smb_trans2 * trans = op - > trans ;
union smb_fsinfo * fsinfo ;
uint16_t level ;
/* make sure we got enough parameters */
if ( trans - > in . params . length ! = 2 ) {
return NT_STATUS_FOOBAR ;
}
fsinfo = talloc ( op , union smb_fsinfo ) ;
NT_STATUS_HAVE_NO_MEMORY ( fsinfo ) ;
level = SVAL ( trans - > in . params . data , 0 ) ;
2006-06-21 23:22:40 +04:00
/* work out the backend level - we make it 1-1 in the header */
fsinfo - > generic . level = ( enum smb_fsinfo_level ) level ;
if ( fsinfo - > generic . level > = RAW_QFS_GENERIC ) {
return NT_STATUS_INVALID_LEVEL ;
2006-03-17 12:42:04 +03:00
}
2006-06-21 23:22:40 +04:00
op - > op_info = fsinfo ;
op - > send_fn = trans2_qfsinfo_send ;
return ntvfs_fsinfo ( req - > ntvfs , fsinfo ) ;
2006-03-17 12:42:04 +03:00
}
/*
trans2 open implementation send
*/
static NTSTATUS trans2_open_send ( struct trans_op * op )
{
struct smbsrv_request * req = op - > req ;
struct smb_trans2 * trans = op - > trans ;
union smb_open * io ;
2006-03-18 14:10:21 +03:00
TRANS2_CHECK_ASYNC_STATUS ( io , union smb_open ) ;
2006-03-17 12:42:04 +03:00
2006-06-20 09:06:10 +04:00
TRANS2_CHECK ( trans2_setup_reply ( trans , 30 , 0 , 0 ) ) ;
2006-03-17 12:42:04 +03:00
2006-05-20 12:15:22 +04:00
smbsrv_push_fnum ( trans - > out . params . data , VWV ( 0 ) , io - > t2open . out . file . ntvfs ) ;
2006-03-17 12:42:04 +03:00
SSVAL ( trans - > out . params . data , VWV ( 1 ) , io - > t2open . out . attrib ) ;
2010-05-01 22:33:20 +04:00
srv_push_dos_date3 ( req - > smb_conn , trans - > out . params . data ,
2006-03-17 12:42:04 +03:00
VWV ( 2 ) , io - > t2open . out . write_time ) ;
SIVAL ( trans - > out . params . data , VWV ( 4 ) , io - > t2open . out . size ) ;
SSVAL ( trans - > out . params . data , VWV ( 6 ) , io - > t2open . out . access ) ;
SSVAL ( trans - > out . params . data , VWV ( 7 ) , io - > t2open . out . ftype ) ;
SSVAL ( trans - > out . params . data , VWV ( 8 ) , io - > t2open . out . devstate ) ;
SSVAL ( trans - > out . params . data , VWV ( 9 ) , io - > t2open . out . action ) ;
SIVAL ( trans - > out . params . data , VWV ( 10 ) , 0 ) ; /* reserved */
SSVAL ( trans - > out . params . data , VWV ( 12 ) , 0 ) ; /* EaErrorOffset */
SIVAL ( trans - > out . params . data , VWV ( 13 ) , 0 ) ; /* EaLength */
return NT_STATUS_OK ;
}
2004-11-06 10:58:45 +03:00
/*
trans2 open implementation
*/
2006-03-17 12:42:04 +03:00
static NTSTATUS trans2_open ( struct smbsrv_request * req , struct trans_op * op )
2004-11-06 10:58:45 +03:00
{
2006-03-17 12:42:04 +03:00
struct smb_trans2 * trans = op - > trans ;
2004-11-06 10:58:45 +03:00
union smb_open * io ;
/* make sure we got enough parameters */
if ( trans - > in . params . length < 29 ) {
return NT_STATUS_FOOBAR ;
}
2006-03-17 12:42:04 +03:00
io = talloc ( op , union smb_open ) ;
NT_STATUS_HAVE_NO_MEMORY ( io ) ;
2004-11-06 10:58:45 +03:00
2005-07-17 16:57:14 +04:00
io - > t2open . level = RAW_OPEN_T2OPEN ;
io - > t2open . in . flags = SVAL ( trans - > in . params . data , VWV ( 0 ) ) ;
io - > t2open . in . open_mode = SVAL ( trans - > in . params . data , VWV ( 1 ) ) ;
io - > t2open . in . search_attrs = SVAL ( trans - > in . params . data , VWV ( 2 ) ) ;
io - > t2open . in . file_attrs = SVAL ( trans - > in . params . data , VWV ( 3 ) ) ;
2010-05-01 22:33:20 +04:00
io - > t2open . in . write_time = srv_pull_dos_date ( req - > smb_conn ,
2006-06-28 19:08:54 +04:00
trans - > in . params . data + VWV ( 4 ) ) ;
2005-07-17 16:57:14 +04:00
io - > t2open . in . open_func = SVAL ( trans - > in . params . data , VWV ( 6 ) ) ;
io - > t2open . in . size = IVAL ( trans - > in . params . data , VWV ( 7 ) ) ;
io - > t2open . in . timeout = IVAL ( trans - > in . params . data , VWV ( 9 ) ) ;
io - > t2open . in . num_eas = 0 ;
io - > t2open . in . eas = NULL ;
2004-11-06 10:58:45 +03:00
2008-02-14 02:12:33 +03:00
smbsrv_blob_pull_string ( & req - > in . bufinfo , & trans - > in . params , 28 , & io - > t2open . in . fname , 0 ) ;
2006-06-20 09:06:10 +04:00
if ( io - > t2open . in . fname = = NULL ) {
return NT_STATUS_FOOBAR ;
}
2004-11-06 10:58:45 +03:00
2006-06-20 09:06:10 +04:00
TRANS2_CHECK ( ea_pull_list ( & trans - > in . data , io , & io - > t2open . in . num_eas , & io - > t2open . in . eas ) ) ;
2004-11-06 10:58:45 +03:00
2006-03-17 12:42:04 +03:00
op - > op_info = io ;
op - > send_fn = trans2_open_send ;
2006-03-18 14:10:21 +03:00
return ntvfs_open ( req - > ntvfs , io ) ;
2006-03-17 12:42:04 +03:00
}
/*
trans2 simple send
*/
static NTSTATUS trans2_simple_send ( struct trans_op * op )
{
struct smbsrv_request * req = op - > req ;
struct smb_trans2 * trans = op - > trans ;
2006-03-18 14:10:21 +03:00
TRANS2_CHECK_ASYNC_STATUS_SIMPLE ;
2004-11-06 10:58:45 +03:00
2006-06-20 09:06:10 +04:00
TRANS2_CHECK ( trans2_setup_reply ( trans , 2 , 0 , 0 ) ) ;
2004-11-06 10:58:45 +03:00
2006-03-17 12:42:04 +03:00
SSVAL ( trans - > out . params . data , VWV ( 0 ) , 0 ) ;
2004-11-06 10:58:45 +03:00
2006-03-17 12:42:04 +03:00
return NT_STATUS_OK ;
2004-11-06 10:58:45 +03:00
}
2004-12-14 02:57:59 +03:00
/*
trans2 mkdir implementation
*/
2006-03-17 12:42:04 +03:00
static NTSTATUS trans2_mkdir ( struct smbsrv_request * req , struct trans_op * op )
2004-12-14 02:57:59 +03:00
{
2006-03-17 12:42:04 +03:00
struct smb_trans2 * trans = op - > trans ;
2004-12-14 02:57:59 +03:00
union smb_mkdir * io ;
/* make sure we got enough parameters */
if ( trans - > in . params . length < 5 ) {
return NT_STATUS_FOOBAR ;
}
2006-03-17 12:42:04 +03:00
io = talloc ( op , union smb_mkdir ) ;
NT_STATUS_HAVE_NO_MEMORY ( io ) ;
2004-12-14 02:57:59 +03:00
io - > t2mkdir . level = RAW_MKDIR_T2MKDIR ;
2008-02-14 02:12:33 +03:00
smbsrv_blob_pull_string ( & req - > in . bufinfo , & trans - > in . params , 4 , & io - > t2mkdir . in . path , 0 ) ;
2006-06-20 09:06:10 +04:00
if ( io - > t2mkdir . in . path = = NULL ) {
return NT_STATUS_FOOBAR ;
}
2004-12-14 02:57:59 +03:00
2010-05-01 22:33:20 +04:00
TRANS2_CHECK ( ea_pull_list ( & trans - > in . data , io ,
& io - > t2mkdir . in . num_eas ,
2006-06-20 09:06:10 +04:00
& io - > t2mkdir . in . eas ) ) ;
2004-12-14 02:57:59 +03:00
2006-03-17 12:42:04 +03:00
op - > op_info = io ;
op - > send_fn = trans2_simple_send ;
2004-12-14 02:57:59 +03:00
2006-03-18 14:10:21 +03:00
return ntvfs_mkdir ( req - > ntvfs , io ) ;
2004-12-14 02:57:59 +03:00
}
2006-06-21 21:22:55 +04:00
static NTSTATUS trans2_push_fileinfo ( struct smbsrv_connection * smb_conn ,
TALLOC_CTX * mem_ctx ,
DATA_BLOB * blob ,
union smb_fileinfo * st ,
int default_str_flags )
{
uint32_t list_size ;
enum smb_fileinfo_level passthru_level ;
switch ( st - > generic . level ) {
case RAW_FILEINFO_GENERIC :
case RAW_FILEINFO_GETATTR :
case RAW_FILEINFO_GETATTRE :
case RAW_FILEINFO_SEC_DESC :
2005-12-06 15:33:32 +03:00
case RAW_FILEINFO_SMB2_ALL_EAS :
case RAW_FILEINFO_SMB2_ALL_INFORMATION :
2006-06-21 21:22:55 +04:00
/* handled elsewhere */
return NT_STATUS_INVALID_LEVEL ;
case RAW_FILEINFO_UNIX_BASIC :
case RAW_FILEINFO_UNIX_LINK :
/* not implemented yet */
2005-12-06 15:33:32 +03:00
return NT_STATUS_INVALID_LEVEL ;
2006-06-21 21:22:55 +04:00
case RAW_FILEINFO_STANDARD :
2006-06-22 01:24:00 +04:00
TRANS2_CHECK ( smbsrv_blob_grow_data ( mem_ctx , blob , 22 ) ) ;
2006-06-21 21:22:55 +04:00
srv_push_dos_date2 ( smb_conn , blob - > data , 0 , st - > standard . out . create_time ) ;
srv_push_dos_date2 ( smb_conn , blob - > data , 4 , st - > standard . out . access_time ) ;
srv_push_dos_date2 ( smb_conn , blob - > data , 8 , st - > standard . out . write_time ) ;
SIVAL ( blob - > data , 12 , st - > standard . out . size ) ;
SIVAL ( blob - > data , 16 , st - > standard . out . alloc_size ) ;
SSVAL ( blob - > data , 20 , st - > standard . out . attrib ) ;
return NT_STATUS_OK ;
case RAW_FILEINFO_EA_SIZE :
2006-06-22 01:24:00 +04:00
TRANS2_CHECK ( smbsrv_blob_grow_data ( mem_ctx , blob , 26 ) ) ;
2006-06-21 21:22:55 +04:00
srv_push_dos_date2 ( smb_conn , blob - > data , 0 , st - > ea_size . out . create_time ) ;
srv_push_dos_date2 ( smb_conn , blob - > data , 4 , st - > ea_size . out . access_time ) ;
srv_push_dos_date2 ( smb_conn , blob - > data , 8 , st - > ea_size . out . write_time ) ;
SIVAL ( blob - > data , 12 , st - > ea_size . out . size ) ;
SIVAL ( blob - > data , 16 , st - > ea_size . out . alloc_size ) ;
SSVAL ( blob - > data , 20 , st - > ea_size . out . attrib ) ;
SIVAL ( blob - > data , 22 , st - > ea_size . out . ea_size ) ;
return NT_STATUS_OK ;
case RAW_FILEINFO_EA_LIST :
list_size = ea_list_size ( st - > ea_list . out . num_eas ,
st - > ea_list . out . eas ) ;
2006-06-22 01:24:00 +04:00
TRANS2_CHECK ( smbsrv_blob_grow_data ( mem_ctx , blob , list_size ) ) ;
2006-06-21 21:22:55 +04:00
2010-05-01 22:33:20 +04:00
ea_put_list ( blob - > data ,
2006-06-21 21:22:55 +04:00
st - > ea_list . out . num_eas , st - > ea_list . out . eas ) ;
return NT_STATUS_OK ;
case RAW_FILEINFO_ALL_EAS :
list_size = ea_list_size ( st - > all_eas . out . num_eas ,
st - > all_eas . out . eas ) ;
2006-06-22 01:24:00 +04:00
TRANS2_CHECK ( smbsrv_blob_grow_data ( mem_ctx , blob , list_size ) ) ;
2006-06-21 21:22:55 +04:00
2010-05-01 22:33:20 +04:00
ea_put_list ( blob - > data ,
2006-06-21 21:22:55 +04:00
st - > all_eas . out . num_eas , st - > all_eas . out . eas ) ;
return NT_STATUS_OK ;
case RAW_FILEINFO_IS_NAME_VALID :
return NT_STATUS_OK ;
case RAW_FILEINFO_BASIC_INFO :
passthru_level = RAW_FILEINFO_BASIC_INFORMATION ;
break ;
case RAW_FILEINFO_STANDARD_INFO :
passthru_level = RAW_FILEINFO_STANDARD_INFORMATION ;
break ;
case RAW_FILEINFO_EA_INFO :
passthru_level = RAW_FILEINFO_EA_INFORMATION ;
break ;
case RAW_FILEINFO_COMPRESSION_INFO :
passthru_level = RAW_FILEINFO_COMPRESSION_INFORMATION ;
break ;
case RAW_FILEINFO_ALL_INFO :
passthru_level = RAW_FILEINFO_ALL_INFORMATION ;
break ;
case RAW_FILEINFO_NAME_INFO :
passthru_level = RAW_FILEINFO_NAME_INFORMATION ;
break ;
case RAW_FILEINFO_ALT_NAME_INFO :
passthru_level = RAW_FILEINFO_ALT_NAME_INFORMATION ;
break ;
case RAW_FILEINFO_STREAM_INFO :
passthru_level = RAW_FILEINFO_STREAM_INFORMATION ;
break ;
default :
passthru_level = st - > generic . level ;
break ;
2003-08-13 05:53:07 +04:00
}
2006-06-22 01:24:00 +04:00
return smbsrv_push_passthru_fileinfo ( mem_ctx , blob ,
2006-06-21 21:22:55 +04:00
passthru_level , st ,
default_str_flags ) ;
2003-08-13 05:53:07 +04:00
}
2006-06-19 22:01:27 +04:00
/*
fill in the reply from a qpathinfo or qfileinfo call
*/
static NTSTATUS trans2_fileinfo_send ( struct trans_op * op )
{
struct smbsrv_request * req = op - > req ;
struct smb_trans2 * trans = op - > trans ;
union smb_fileinfo * st ;
TRANS2_CHECK_ASYNC_STATUS ( st , union smb_fileinfo ) ;
TRANS2_CHECK ( trans2_setup_reply ( trans , 2 , 0 , 0 ) ) ;
SSVAL ( trans - > out . params . data , 0 , 0 ) ;
2006-06-20 10:32:29 +04:00
TRANS2_CHECK ( trans2_push_fileinfo ( req - > smb_conn , trans ,
& trans - > out . data , st ,
2006-06-22 01:24:00 +04:00
SMBSRV_REQ_DEFAULT_STR_FLAGS ( req ) ) ) ;
2006-06-19 22:01:27 +04:00
return NT_STATUS_OK ;
}
2003-08-13 05:53:07 +04:00
/*
trans2 qpathinfo implementation
*/
2006-03-17 12:42:04 +03:00
static NTSTATUS trans2_qpathinfo ( struct smbsrv_request * req , struct trans_op * op )
2003-08-13 05:53:07 +04:00
{
2006-03-17 12:42:04 +03:00
struct smb_trans2 * trans = op - > trans ;
union smb_fileinfo * st ;
2004-05-25 21:24:24 +04:00
uint16_t level ;
2003-08-13 05:53:07 +04:00
/* make sure we got enough parameters */
2004-10-02 16:16:46 +04:00
if ( trans - > in . params . length < 2 ) {
2003-08-13 05:53:07 +04:00
return NT_STATUS_FOOBAR ;
}
2006-03-17 12:42:04 +03:00
st = talloc ( op , union smb_fileinfo ) ;
NT_STATUS_HAVE_NO_MEMORY ( st ) ;
2003-08-13 05:53:07 +04:00
level = SVAL ( trans - > in . params . data , 0 ) ;
2008-02-14 02:12:33 +03:00
smbsrv_blob_pull_string ( & req - > in . bufinfo , & trans - > in . params , 6 , & st - > generic . in . file . path , 0 ) ;
2006-03-17 12:42:04 +03:00
if ( st - > generic . in . file . path = = NULL ) {
2003-08-13 05:53:07 +04:00
return NT_STATUS_FOOBAR ;
}
/* work out the backend level - we make it 1-1 in the header */
2006-03-17 12:42:04 +03:00
st - > generic . level = ( enum smb_fileinfo_level ) level ;
if ( st - > generic . level > = RAW_FILEINFO_GENERIC ) {
2003-08-13 05:53:07 +04:00
return NT_STATUS_INVALID_LEVEL ;
}
2006-03-17 12:42:04 +03:00
if ( st - > generic . level = = RAW_FILEINFO_EA_LIST ) {
2010-05-01 22:33:20 +04:00
TRANS2_CHECK ( ea_pull_name_list ( & trans - > in . data , req ,
2006-06-20 09:06:10 +04:00
& st - > ea_list . in . num_names ,
& st - > ea_list . in . ea_names ) ) ;
2003-08-13 05:53:07 +04:00
}
2006-03-17 12:42:04 +03:00
op - > op_info = st ;
op - > send_fn = trans2_fileinfo_send ;
2003-08-13 05:53:07 +04:00
2006-03-18 14:10:21 +03:00
return ntvfs_qpathinfo ( req - > ntvfs , st ) ;
2003-08-13 05:53:07 +04:00
}
/*
trans2 qpathinfo implementation
*/
2006-03-17 12:42:04 +03:00
static NTSTATUS trans2_qfileinfo ( struct smbsrv_request * req , struct trans_op * op )
2003-08-13 05:53:07 +04:00
{
2006-03-17 12:42:04 +03:00
struct smb_trans2 * trans = op - > trans ;
union smb_fileinfo * st ;
2006-05-20 12:15:22 +04:00
uint16_t level ;
struct ntvfs_handle * h ;
2003-08-13 05:53:07 +04:00
/* make sure we got enough parameters */
if ( trans - > in . params . length < 4 ) {
return NT_STATUS_FOOBAR ;
}
2006-03-17 12:42:04 +03:00
st = talloc ( op , union smb_fileinfo ) ;
NT_STATUS_HAVE_NO_MEMORY ( st ) ;
2006-05-20 12:15:22 +04:00
h = smbsrv_pull_fnum ( req , trans - > in . params . data , 0 ) ;
2003-08-13 05:53:07 +04:00
level = SVAL ( trans - > in . params . data , 2 ) ;
2006-05-20 12:15:22 +04:00
st - > generic . in . file . ntvfs = h ;
2003-08-13 05:53:07 +04:00
/* work out the backend level - we make it 1-1 in the header */
2006-03-17 12:42:04 +03:00
st - > generic . level = ( enum smb_fileinfo_level ) level ;
if ( st - > generic . level > = RAW_FILEINFO_GENERIC ) {
2003-08-13 05:53:07 +04:00
return NT_STATUS_INVALID_LEVEL ;
}
2006-03-17 12:42:04 +03:00
if ( st - > generic . level = = RAW_FILEINFO_EA_LIST ) {
2010-05-01 22:33:20 +04:00
TRANS2_CHECK ( ea_pull_name_list ( & trans - > in . data , req ,
2006-06-20 09:06:10 +04:00
& st - > ea_list . in . num_names ,
& st - > ea_list . in . ea_names ) ) ;
2004-12-18 01:47:49 +03:00
}
2006-03-17 12:42:04 +03:00
op - > op_info = st ;
op - > send_fn = trans2_fileinfo_send ;
2003-08-13 05:53:07 +04:00
2006-05-20 12:15:22 +04:00
SMBSRV_CHECK_FILE_HANDLE_NTSTATUS ( st - > generic . in . file . ntvfs ) ;
2006-03-18 14:10:21 +03:00
return ntvfs_qfileinfo ( req - > ntvfs , st ) ;
2003-08-13 05:53:07 +04:00
}
/*
parse a trans2 setfileinfo / setpathinfo data blob
*/
2004-06-28 12:39:00 +04:00
static NTSTATUS trans2_parse_sfileinfo ( struct smbsrv_request * req ,
2003-08-13 05:53:07 +04:00
union smb_setfileinfo * st ,
const DATA_BLOB * blob )
{
2006-06-28 19:08:54 +04:00
enum smb_setfileinfo_level passthru_level ;
2003-08-13 05:53:07 +04:00
switch ( st - > generic . level ) {
case RAW_SFILEINFO_GENERIC :
case RAW_SFILEINFO_SETATTR :
case RAW_SFILEINFO_SETATTRE :
2004-11-18 04:41:43 +03:00
case RAW_SFILEINFO_SEC_DESC :
2003-08-13 05:53:07 +04:00
/* handled elsewhere */
return NT_STATUS_INVALID_LEVEL ;
case RAW_SFILEINFO_STANDARD :
CHECK_MIN_BLOB_SIZE ( blob , 12 ) ;
2006-06-20 09:06:10 +04:00
2004-06-29 11:40:14 +04:00
st - > standard . in . create_time = srv_pull_dos_date2 ( req - > smb_conn , blob - > data + 0 ) ;
st - > standard . in . access_time = srv_pull_dos_date2 ( req - > smb_conn , blob - > data + 4 ) ;
st - > standard . in . write_time = srv_pull_dos_date2 ( req - > smb_conn , blob - > data + 8 ) ;
2006-06-20 09:06:10 +04:00
2003-08-13 05:53:07 +04:00
return NT_STATUS_OK ;
case RAW_SFILEINFO_EA_SET :
2010-05-01 22:33:20 +04:00
return ea_pull_list ( blob , req ,
& st - > ea_set . in . num_eas ,
2004-12-17 07:51:23 +03:00
& st - > ea_set . in . eas ) ;
2003-08-13 05:53:07 +04:00
case SMB_SFILEINFO_BASIC_INFO :
case SMB_SFILEINFO_BASIC_INFORMATION :
2006-06-28 19:08:54 +04:00
passthru_level = SMB_SFILEINFO_BASIC_INFORMATION ;
break ;
2003-08-13 05:53:07 +04:00
case SMB_SFILEINFO_DISPOSITION_INFO :
case SMB_SFILEINFO_DISPOSITION_INFORMATION :
2006-06-28 19:08:54 +04:00
passthru_level = SMB_SFILEINFO_DISPOSITION_INFORMATION ;
break ;
2003-08-13 05:53:07 +04:00
case SMB_SFILEINFO_ALLOCATION_INFO :
case SMB_SFILEINFO_ALLOCATION_INFORMATION :
2006-06-28 19:08:54 +04:00
passthru_level = SMB_SFILEINFO_ALLOCATION_INFORMATION ;
break ;
2003-08-13 05:53:07 +04:00
case RAW_SFILEINFO_END_OF_FILE_INFO :
case RAW_SFILEINFO_END_OF_FILE_INFORMATION :
2006-06-28 19:08:54 +04:00
passthru_level = RAW_SFILEINFO_END_OF_FILE_INFORMATION ;
break ;
2003-08-13 05:53:07 +04:00
2006-06-20 09:06:10 +04:00
case RAW_SFILEINFO_RENAME_INFORMATION :
2003-08-13 05:53:07 +04:00
case RAW_SFILEINFO_POSITION_INFORMATION :
case RAW_SFILEINFO_MODE_INFORMATION :
2006-06-28 19:08:54 +04:00
passthru_level = st - > generic . level ;
break ;
2005-09-16 11:15:40 +04:00
case RAW_SFILEINFO_UNIX_BASIC :
case RAW_SFILEINFO_UNIX_LINK :
case RAW_SFILEINFO_UNIX_HLINK :
2008-06-02 05:05:06 +04:00
case RAW_SFILEINFO_PIPE_INFORMATION :
case RAW_SFILEINFO_VALID_DATA_INFORMATION :
case RAW_SFILEINFO_SHORT_NAME_INFORMATION :
2005-09-16 11:15:40 +04:00
case RAW_SFILEINFO_1025 :
2008-06-02 05:05:06 +04:00
case RAW_SFILEINFO_1027 :
2005-09-16 11:15:40 +04:00
case RAW_SFILEINFO_1029 :
2008-06-02 05:05:06 +04:00
case RAW_SFILEINFO_1030 :
case RAW_SFILEINFO_1031 :
2005-09-16 11:15:40 +04:00
case RAW_SFILEINFO_1032 :
2008-06-02 05:05:06 +04:00
case RAW_SFILEINFO_1036 :
case RAW_SFILEINFO_1041 :
case RAW_SFILEINFO_1042 :
case RAW_SFILEINFO_1043 :
case RAW_SFILEINFO_1044 :
2005-09-16 11:15:40 +04:00
return NT_STATUS_INVALID_LEVEL ;
2006-10-28 13:40:00 +04:00
default :
/* we need a default here to cope with invalid values on the wire */
return NT_STATUS_INVALID_LEVEL ;
2003-08-13 05:53:07 +04:00
}
2006-06-28 19:08:54 +04:00
return smbsrv_pull_passthru_sfileinfo ( st , passthru_level , st ,
blob , SMBSRV_REQ_DEFAULT_STR_FLAGS ( req ) ,
2008-02-14 02:12:33 +03:00
& req - > in . bufinfo ) ;
2003-08-13 05:53:07 +04:00
}
/*
trans2 setfileinfo implementation
*/
2006-03-17 12:42:04 +03:00
static NTSTATUS trans2_setfileinfo ( struct smbsrv_request * req , struct trans_op * op )
2003-08-13 05:53:07 +04:00
{
2006-03-17 12:42:04 +03:00
struct smb_trans2 * trans = op - > trans ;
union smb_setfileinfo * st ;
2006-05-20 12:15:22 +04:00
uint16_t level ;
struct ntvfs_handle * h ;
2003-08-13 05:53:07 +04:00
/* make sure we got enough parameters */
if ( trans - > in . params . length < 4 ) {
return NT_STATUS_FOOBAR ;
}
2006-03-17 12:42:04 +03:00
st = talloc ( op , union smb_setfileinfo ) ;
NT_STATUS_HAVE_NO_MEMORY ( st ) ;
2006-05-20 12:15:22 +04:00
h = smbsrv_pull_fnum ( req , trans - > in . params . data , 0 ) ;
2003-08-13 05:53:07 +04:00
level = SVAL ( trans - > in . params . data , 2 ) ;
2006-05-20 12:15:22 +04:00
st - > generic . in . file . ntvfs = h ;
2006-03-17 12:42:04 +03:00
/* work out the backend level - we make it 1-1 in the header */
st - > generic . level = ( enum smb_setfileinfo_level ) level ;
if ( st - > generic . level > = RAW_SFILEINFO_GENERIC ) {
return NT_STATUS_INVALID_LEVEL ;
2003-08-13 05:53:07 +04:00
}
2006-06-20 09:06:10 +04:00
TRANS2_CHECK ( trans2_parse_sfileinfo ( req , st , & trans - > in . data ) ) ;
2003-08-13 05:53:07 +04:00
2006-03-17 12:42:04 +03:00
op - > op_info = st ;
op - > send_fn = trans2_simple_send ;
2006-05-20 12:15:22 +04:00
SMBSRV_CHECK_FILE_HANDLE_NTSTATUS ( st - > generic . in . file . ntvfs ) ;
2006-03-18 14:10:21 +03:00
return ntvfs_setfileinfo ( req - > ntvfs , st ) ;
2003-08-13 05:53:07 +04:00
}
/*
trans2 setpathinfo implementation
*/
2006-03-17 12:42:04 +03:00
static NTSTATUS trans2_setpathinfo ( struct smbsrv_request * req , struct trans_op * op )
2003-08-13 05:53:07 +04:00
{
2006-03-17 12:42:04 +03:00
struct smb_trans2 * trans = op - > trans ;
union smb_setfileinfo * st ;
2004-05-25 21:24:24 +04:00
uint16_t level ;
2003-08-13 05:53:07 +04:00
/* make sure we got enough parameters */
if ( trans - > in . params . length < 4 ) {
return NT_STATUS_FOOBAR ;
}
2006-03-17 12:42:04 +03:00
st = talloc ( op , union smb_setfileinfo ) ;
NT_STATUS_HAVE_NO_MEMORY ( st ) ;
2003-08-13 05:53:07 +04:00
level = SVAL ( trans - > in . params . data , 0 ) ;
2008-02-14 02:12:33 +03:00
smbsrv_blob_pull_string ( & req - > in . bufinfo , & trans - > in . params , 6 , & st - > generic . in . file . path , 0 ) ;
2006-03-17 12:42:04 +03:00
if ( st - > generic . in . file . path = = NULL ) {
2003-08-13 05:53:07 +04:00
return NT_STATUS_FOOBAR ;
}
2006-03-17 12:42:04 +03:00
/* work out the backend level - we make it 1-1 in the header */
st - > generic . level = ( enum smb_setfileinfo_level ) level ;
if ( st - > generic . level > = RAW_SFILEINFO_GENERIC ) {
return NT_STATUS_INVALID_LEVEL ;
2003-08-13 05:53:07 +04:00
}
2006-06-20 09:06:10 +04:00
TRANS2_CHECK ( trans2_parse_sfileinfo ( req , st , & trans - > in . data ) ) ;
2003-08-13 05:53:07 +04:00
2006-03-17 12:42:04 +03:00
op - > op_info = st ;
op - > send_fn = trans2_simple_send ;
2006-03-18 14:10:21 +03:00
return ntvfs_setpathinfo ( req - > ntvfs , st ) ;
2003-08-13 05:53:07 +04:00
}
/* a structure to encapsulate the state information about an in-progress ffirst/fnext operation */
struct find_state {
2006-03-17 12:42:04 +03:00
struct trans_op * op ;
void * search ;
2006-07-06 12:00:24 +04:00
enum smb_search_data_level data_level ;
2004-05-25 21:24:24 +04:00
uint16_t last_entry_offset ;
uint16_t flags ;
2003-08-13 05:53:07 +04:00
} ;
/*
2010-05-01 22:33:20 +04:00
fill a single entry in a trans2 find reply
2003-08-13 05:53:07 +04:00
*/
2006-06-22 00:43:59 +04:00
static NTSTATUS find_fill_info ( struct find_state * state ,
2007-05-22 13:25:58 +04:00
const union smb_search_data * file )
2003-08-13 05:53:07 +04:00
{
2006-03-17 12:42:04 +03:00
struct smbsrv_request * req = state - > op - > req ;
struct smb_trans2 * trans = state - > op - > trans ;
2004-11-25 23:01:47 +03:00
uint8_t * data ;
2010-01-05 20:42:54 +03:00
unsigned int ofs = trans - > out . data . length ;
2004-12-18 07:38:43 +03:00
uint32_t ea_size ;
2003-08-13 05:53:07 +04:00
2006-07-06 12:00:24 +04:00
switch ( state - > data_level ) {
case RAW_SEARCH_DATA_GENERIC :
case RAW_SEARCH_DATA_SEARCH :
2003-08-13 05:53:07 +04:00
/* handled elsewhere */
2006-06-22 00:43:59 +04:00
return NT_STATUS_INVALID_LEVEL ;
2003-08-13 05:53:07 +04:00
2006-07-06 12:00:24 +04:00
case RAW_SEARCH_DATA_STANDARD :
2003-08-31 07:16:52 +04:00
if ( state - > flags & FLAG_TRANS2_FIND_REQUIRE_RESUME ) {
2006-06-22 01:24:00 +04:00
TRANS2_CHECK ( smbsrv_blob_grow_data ( trans , & trans - > out . data , ofs + 27 ) ) ;
2003-08-31 07:16:52 +04:00
SIVAL ( trans - > out . data . data , ofs , file - > standard . resume_key ) ;
ofs + = 4 ;
} else {
2006-06-22 01:24:00 +04:00
TRANS2_CHECK ( smbsrv_blob_grow_data ( trans , & trans - > out . data , ofs + 23 ) ) ;
2003-08-31 07:16:52 +04:00
}
2003-08-13 05:53:07 +04:00
data = trans - > out . data . data + ofs ;
2004-06-29 11:40:14 +04:00
srv_push_dos_date2 ( req - > smb_conn , data , 0 , file - > standard . create_time ) ;
srv_push_dos_date2 ( req - > smb_conn , data , 4 , file - > standard . access_time ) ;
srv_push_dos_date2 ( req - > smb_conn , data , 8 , file - > standard . write_time ) ;
2003-08-13 05:53:07 +04:00
SIVAL ( data , 12 , file - > standard . size ) ;
SIVAL ( data , 16 , file - > standard . alloc_size ) ;
SSVAL ( data , 20 , file - > standard . attrib ) ;
2010-05-01 22:33:20 +04:00
TRANS2_CHECK ( smbsrv_blob_append_string ( trans , & trans - > out . data , file - > standard . name . s ,
2006-06-22 01:24:00 +04:00
ofs + 22 , SMBSRV_REQ_DEFAULT_STR_FLAGS ( req ) ,
2006-06-22 00:43:59 +04:00
STR_LEN8BIT | STR_TERMINATE | STR_LEN_NOTERM ) ) ;
2003-08-13 05:53:07 +04:00
break ;
2006-07-06 12:00:24 +04:00
case RAW_SEARCH_DATA_EA_SIZE :
2003-08-31 07:16:52 +04:00
if ( state - > flags & FLAG_TRANS2_FIND_REQUIRE_RESUME ) {
2006-06-22 01:24:00 +04:00
TRANS2_CHECK ( smbsrv_blob_grow_data ( trans , & trans - > out . data , ofs + 31 ) ) ;
2003-08-31 07:16:52 +04:00
SIVAL ( trans - > out . data . data , ofs , file - > ea_size . resume_key ) ;
ofs + = 4 ;
} else {
2006-06-22 01:24:00 +04:00
TRANS2_CHECK ( smbsrv_blob_grow_data ( trans , & trans - > out . data , ofs + 27 ) ) ;
2003-08-31 07:16:52 +04:00
}
2003-08-13 05:53:07 +04:00
data = trans - > out . data . data + ofs ;
2004-06-29 11:40:14 +04:00
srv_push_dos_date2 ( req - > smb_conn , data , 0 , file - > ea_size . create_time ) ;
srv_push_dos_date2 ( req - > smb_conn , data , 4 , file - > ea_size . access_time ) ;
srv_push_dos_date2 ( req - > smb_conn , data , 8 , file - > ea_size . write_time ) ;
2003-08-13 05:53:07 +04:00
SIVAL ( data , 12 , file - > ea_size . size ) ;
SIVAL ( data , 16 , file - > ea_size . alloc_size ) ;
SSVAL ( data , 20 , file - > ea_size . attrib ) ;
SIVAL ( data , 22 , file - > ea_size . ea_size ) ;
2010-05-01 22:33:20 +04:00
TRANS2_CHECK ( smbsrv_blob_append_string ( trans , & trans - > out . data , file - > ea_size . name . s ,
2006-06-22 01:24:00 +04:00
ofs + 26 , SMBSRV_REQ_DEFAULT_STR_FLAGS ( req ) ,
2006-06-22 00:43:59 +04:00
STR_LEN8BIT | STR_NOALIGN ) ) ;
2006-06-22 01:24:00 +04:00
TRANS2_CHECK ( smbsrv_blob_fill_data ( trans , & trans - > out . data , trans - > out . data . length + 1 ) ) ;
2003-08-13 05:53:07 +04:00
break ;
2006-07-06 12:00:24 +04:00
case RAW_SEARCH_DATA_EA_LIST :
2004-12-18 07:38:43 +03:00
ea_size = ea_list_size ( file - > ea_list . eas . num_eas , file - > ea_list . eas . eas ) ;
if ( state - > flags & FLAG_TRANS2_FIND_REQUIRE_RESUME ) {
2006-06-22 01:24:00 +04:00
TRANS2_CHECK ( smbsrv_blob_grow_data ( trans , & trans - > out . data , ofs + 27 + ea_size ) ) ;
2004-12-18 07:38:43 +03:00
SIVAL ( trans - > out . data . data , ofs , file - > ea_list . resume_key ) ;
ofs + = 4 ;
} else {
2006-06-22 01:24:00 +04:00
TRANS2_CHECK ( smbsrv_blob_grow_data ( trans , & trans - > out . data , ofs + 23 + ea_size ) ) ;
2004-12-18 07:38:43 +03:00
}
data = trans - > out . data . data + ofs ;
srv_push_dos_date2 ( req - > smb_conn , data , 0 , file - > ea_list . create_time ) ;
srv_push_dos_date2 ( req - > smb_conn , data , 4 , file - > ea_list . access_time ) ;
srv_push_dos_date2 ( req - > smb_conn , data , 8 , file - > ea_list . write_time ) ;
SIVAL ( data , 12 , file - > ea_list . size ) ;
SIVAL ( data , 16 , file - > ea_list . alloc_size ) ;
SSVAL ( data , 20 , file - > ea_list . attrib ) ;
ea_put_list ( data + 22 , file - > ea_list . eas . num_eas , file - > ea_list . eas . eas ) ;
2010-05-01 22:33:20 +04:00
TRANS2_CHECK ( smbsrv_blob_append_string ( trans , & trans - > out . data , file - > ea_list . name . s ,
2006-06-22 01:24:00 +04:00
ofs + 22 + ea_size , SMBSRV_REQ_DEFAULT_STR_FLAGS ( req ) ,
2006-06-22 00:43:59 +04:00
STR_LEN8BIT | STR_NOALIGN ) ) ;
2006-06-22 01:24:00 +04:00
TRANS2_CHECK ( smbsrv_blob_fill_data ( trans , & trans - > out . data , trans - > out . data . length + 1 ) ) ;
2004-12-18 07:38:43 +03:00
break ;
2006-07-06 12:00:24 +04:00
case RAW_SEARCH_DATA_DIRECTORY_INFO :
case RAW_SEARCH_DATA_FULL_DIRECTORY_INFO :
case RAW_SEARCH_DATA_NAME_INFO :
case RAW_SEARCH_DATA_BOTH_DIRECTORY_INFO :
case RAW_SEARCH_DATA_ID_FULL_DIRECTORY_INFO :
case RAW_SEARCH_DATA_ID_BOTH_DIRECTORY_INFO :
return smbsrv_push_passthru_search ( trans , & trans - > out . data , state - > data_level , file ,
2006-07-05 10:47:34 +04:00
SMBSRV_REQ_DEFAULT_STR_FLAGS ( req ) ) ;
2006-07-06 12:00:24 +04:00
case RAW_SEARCH_DATA_UNIX_INFO :
2008-06-02 05:05:06 +04:00
case RAW_SEARCH_DATA_UNIX_INFO2 :
2006-07-06 12:00:24 +04:00
return NT_STATUS_INVALID_LEVEL ;
2003-08-13 05:53:07 +04:00
}
2004-12-20 06:12:09 +03:00
2006-06-22 00:43:59 +04:00
return NT_STATUS_OK ;
2003-08-13 05:53:07 +04:00
}
/* callback function for trans2 findfirst/findnext */
2009-02-02 10:39:45 +03:00
static bool find_callback ( void * private_data , const union smb_search_data * file )
2003-08-13 05:53:07 +04:00
{
2009-02-02 10:39:45 +03:00
struct find_state * state = talloc_get_type ( private_data , struct find_state ) ;
2006-03-17 12:42:04 +03:00
struct smb_trans2 * trans = state - > op - > trans ;
2010-01-05 20:42:54 +03:00
unsigned int old_length ;
2003-08-13 05:53:07 +04:00
old_length = trans - > out . data . length ;
2006-06-22 00:43:59 +04:00
if ( ! NT_STATUS_IS_OK ( find_fill_info ( state , file ) ) | |
2004-12-20 06:12:09 +03:00
trans - > out . data . length > trans - > in . max_data ) {
2003-08-13 05:53:07 +04:00
/* restore the old length and tell the backend to stop */
2006-06-22 01:24:00 +04:00
smbsrv_blob_grow_data ( trans , & trans - > out . data , old_length ) ;
2007-10-07 02:10:49 +04:00
return false ;
2003-08-13 05:53:07 +04:00
}
state - > last_entry_offset = old_length ;
2007-10-07 02:10:49 +04:00
return true ;
2003-08-13 05:53:07 +04:00
}
2006-03-17 12:42:04 +03:00
/*
trans2 findfirst send
*/
static NTSTATUS trans2_findfirst_send ( struct trans_op * op )
{
struct smbsrv_request * req = op - > req ;
struct smb_trans2 * trans = op - > trans ;
union smb_search_first * search ;
struct find_state * state ;
uint8_t * param ;
2006-03-18 14:10:21 +03:00
TRANS2_CHECK_ASYNC_STATUS ( state , struct find_state ) ;
2006-03-17 12:42:04 +03:00
search = talloc_get_type ( state - > search , union smb_search_first ) ;
/* fill in the findfirst reply header */
param = trans - > out . params . data ;
SSVAL ( param , VWV ( 0 ) , search - > t2ffirst . out . handle ) ;
SSVAL ( param , VWV ( 1 ) , search - > t2ffirst . out . count ) ;
SSVAL ( param , VWV ( 2 ) , search - > t2ffirst . out . end_of_search ) ;
SSVAL ( param , VWV ( 3 ) , 0 ) ;
SSVAL ( param , VWV ( 4 ) , state - > last_entry_offset ) ;
return NT_STATUS_OK ;
}
2010-05-01 22:27:31 +04:00
/*
2011-10-04 14:42:11 +04:00
trans2 getdfsreferral implementation
*/
static NTSTATUS trans2_getdfsreferral ( struct smbsrv_request * req ,
struct trans_op * op )
2010-05-01 22:27:31 +04:00
{
2011-10-04 14:42:11 +04:00
enum ndr_err_code ndr_err ;
struct smb_trans2 * trans = op - > trans ;
struct ldb_context * ldb ;
struct loadparm_context * lp_ctx ;
2010-05-01 22:27:31 +04:00
NTSTATUS status ;
2011-10-04 14:42:11 +04:00
struct dfs_GetDFSReferral * r ;
DATA_BLOB outblob = data_blob_null ;
uint16_t nb_referrals = 0 ;
2010-05-01 22:27:31 +04:00
2011-10-04 14:42:11 +04:00
lp_ctx = req - > tcon - > ntvfs - > lp_ctx ;
if ( ! lpcfg_host_msdfs ( lp_ctx ) ) {
return NT_STATUS_NOT_IMPLEMENTED ;
2010-05-01 22:27:31 +04:00
}
2011-10-04 14:42:11 +04:00
r = talloc_zero ( req , struct dfs_GetDFSReferral ) ;
NT_STATUS_HAVE_NO_MEMORY ( r ) ;
2010-05-01 22:27:31 +04:00
2018-04-11 21:41:30 +03:00
ldb = samdb_connect ( r ,
req - > tcon - > ntvfs - > event_ctx ,
lp_ctx ,
system_session ( lp_ctx ) ,
NULL ,
0 ) ;
2011-10-04 14:42:11 +04:00
if ( ldb = = NULL ) {
DEBUG ( 2 , ( __location__ " : Failed to open samdb \n " ) ) ;
talloc_free ( r ) ;
2010-05-01 22:27:31 +04:00
return NT_STATUS_INTERNAL_ERROR ;
}
2011-10-04 14:42:11 +04:00
ndr_err = ndr_pull_struct_blob ( & trans - > in . params , r ,
& r - > in . req ,
( ndr_pull_flags_fn_t ) ndr_pull_dfs_GetDFSReferral_in ) ;
if ( ! NDR_ERR_CODE_IS_SUCCESS ( ndr_err ) ) {
status = ndr_map_error2ntstatus ( ndr_err ) ;
DEBUG ( 2 , ( __location__ " : Failed to parse GetDFSReferral_in - %s \n " ,
nt_errstr ( status ) ) ) ;
2010-05-01 22:27:31 +04:00
talloc_free ( r ) ;
2011-10-04 14:42:11 +04:00
return status ;
2010-05-01 22:27:31 +04:00
}
2011-10-04 14:42:11 +04:00
DEBUG ( 8 , ( " Requested DFS name: %s length: %u \n " ,
r - > in . req . servername ,
( unsigned int ) strlen_m ( r - > in . req . servername ) * 2 ) ) ;
2010-05-01 22:27:31 +04:00
2011-10-04 14:42:11 +04:00
status = dfs_server_ad_get_referrals ( lp_ctx , ldb ,
req - > smb_conn - > connection - > remote_address , r ) ;
2010-05-01 22:27:31 +04:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2011-10-04 14:42:11 +04:00
talloc_free ( r ) ;
return status ;
2010-05-01 22:27:31 +04:00
}
2011-10-04 14:42:11 +04:00
ndr_err = ndr_push_struct_blob ( & outblob , trans ,
r - > out . resp ,
2010-05-01 22:27:31 +04:00
( ndr_push_flags_fn_t ) ndr_push_dfs_referral_resp ) ;
if ( ! NDR_ERR_CODE_IS_SUCCESS ( ndr_err ) ) {
2023-08-01 06:52:07 +03:00
DEBUG ( 2 , ( __location__ " :NDR marshalling of domain referral response failed \n " ) ) ;
2011-10-04 14:42:11 +04:00
talloc_free ( r ) ;
2010-05-01 22:27:31 +04:00
return NT_STATUS_INTERNAL_ERROR ;
}
2011-10-04 14:42:11 +04:00
nb_referrals = r - > out . resp - > nb_referrals ;
2010-05-01 22:27:31 +04:00
if ( outblob . length > trans - > in . max_data ) {
bool ok = false ;
DEBUG ( 3 , ( " Blob is too big for the output buffer "
" size %u max %u \n " ,
( unsigned int ) outblob . length , trans - > in . max_data ) ) ;
if ( trans - > in . max_data ! = MAX_DFS_RESPONSE ) {
/* As specified in MS-DFSC.pdf 3.3.5.2 */
2011-10-04 14:42:11 +04:00
talloc_free ( r ) ;
2010-05-01 22:27:31 +04:00
return STATUS_BUFFER_OVERFLOW ;
}
/*
* The answer is too big , so let ' s remove some answers
*/
2011-10-04 14:42:11 +04:00
while ( ! ok & & r - > out . resp - > nb_referrals > 2 ) {
2010-05-01 22:27:31 +04:00
data_blob_free ( & outblob ) ;
/*
2011-10-04 14:42:11 +04:00
* Let ' s scrap the last referral ( for now )
2010-05-01 22:27:31 +04:00
*/
2011-10-04 14:42:11 +04:00
r - > out . resp - > nb_referrals - = 1 ;
2010-05-01 22:27:31 +04:00
2011-10-04 14:42:11 +04:00
ndr_err = ndr_push_struct_blob ( & outblob , trans ,
r - > out . resp ,
2010-05-01 22:27:31 +04:00
( ndr_push_flags_fn_t ) ndr_push_dfs_referral_resp ) ;
if ( ! NDR_ERR_CODE_IS_SUCCESS ( ndr_err ) ) {
2011-10-04 14:42:11 +04:00
talloc_free ( r ) ;
2010-05-01 22:27:31 +04:00
return NT_STATUS_INTERNAL_ERROR ;
}
if ( outblob . length < = MAX_DFS_RESPONSE ) {
2023-08-07 07:54:43 +03:00
DEBUG ( 10 , ( " DFS: managed to reduce the size of referral initial "
" number of referral %d, actual count: %d \n " ,
2011-10-04 14:42:11 +04:00
nb_referrals , r - > out . resp - > nb_referrals ) ) ;
2010-05-01 22:27:31 +04:00
ok = true ;
break ;
}
}
2011-10-04 14:42:11 +04:00
if ( ! ok & & r - > out . resp - > nb_referrals < = 2 ) {
2011-05-15 22:18:19 +04:00
DEBUG ( 8 , ( __location__ " ; Not able to fit the domain and realm in DFS a "
2023-08-07 07:54:43 +03:00
" 56K buffer, something must be broken \n " ) ) ;
2011-10-04 14:42:11 +04:00
talloc_free ( r ) ;
2010-05-01 22:27:31 +04:00
return NT_STATUS_INTERNAL_ERROR ;
}
}
TRANS2_CHECK ( trans2_setup_reply ( trans , 0 , outblob . length , 0 ) ) ;
trans - > out . data = outblob ;
2011-10-04 14:42:11 +04:00
talloc_free ( r ) ;
2010-05-01 22:27:31 +04:00
return NT_STATUS_OK ;
}
2003-08-13 05:53:07 +04:00
/*
trans2 findfirst implementation
*/
2006-03-17 12:42:04 +03:00
static NTSTATUS trans2_findfirst ( struct smbsrv_request * req , struct trans_op * op )
2003-08-13 05:53:07 +04:00
{
2006-03-17 12:42:04 +03:00
struct smb_trans2 * trans = op - > trans ;
union smb_search_first * search ;
2004-05-25 21:24:24 +04:00
uint16_t level ;
2006-03-17 12:42:04 +03:00
struct find_state * state ;
2003-08-13 05:53:07 +04:00
/* make sure we got all the parameters */
if ( trans - > in . params . length < 14 ) {
return NT_STATUS_FOOBAR ;
}
2006-03-17 12:42:04 +03:00
search = talloc ( op , union smb_search_first ) ;
NT_STATUS_HAVE_NO_MEMORY ( search ) ;
2003-08-13 05:53:07 +04:00
2006-03-17 12:42:04 +03:00
search - > t2ffirst . in . search_attrib = SVAL ( trans - > in . params . data , 0 ) ;
search - > t2ffirst . in . max_count = SVAL ( trans - > in . params . data , 2 ) ;
search - > t2ffirst . in . flags = SVAL ( trans - > in . params . data , 4 ) ;
level = SVAL ( trans - > in . params . data , 6 ) ;
search - > t2ffirst . in . storage_type = IVAL ( trans - > in . params . data , 8 ) ;
2008-02-14 02:12:33 +03:00
smbsrv_blob_pull_string ( & req - > in . bufinfo , & trans - > in . params , 12 , & search - > t2ffirst . in . pattern , 0 ) ;
2006-03-17 12:42:04 +03:00
if ( search - > t2ffirst . in . pattern = = NULL ) {
2003-08-13 05:53:07 +04:00
return NT_STATUS_FOOBAR ;
}
2006-07-06 12:00:24 +04:00
search - > t2ffirst . level = RAW_SEARCH_TRANS2 ;
search - > t2ffirst . data_level = ( enum smb_search_data_level ) level ;
if ( search - > t2ffirst . data_level > = RAW_SEARCH_DATA_GENERIC ) {
2003-08-13 05:53:07 +04:00
return NT_STATUS_INVALID_LEVEL ;
}
2006-07-06 12:00:24 +04:00
if ( search - > t2ffirst . data_level = = RAW_SEARCH_DATA_EA_LIST ) {
2006-06-20 09:06:10 +04:00
TRANS2_CHECK ( ea_pull_name_list ( & trans - > in . data , req ,
2010-05-01 22:33:20 +04:00
& search - > t2ffirst . in . num_names ,
2006-06-20 09:06:10 +04:00
& search - > t2ffirst . in . ea_names ) ) ;
2004-12-18 07:38:43 +03:00
}
2004-12-29 02:28:02 +03:00
/* setup the private state structure that the backend will
give us in the callback */
2006-03-17 12:42:04 +03:00
state = talloc ( op , struct find_state ) ;
NT_STATUS_HAVE_NO_MEMORY ( state ) ;
state - > op = op ;
state - > search = search ;
2006-07-06 12:00:24 +04:00
state - > data_level = search - > t2ffirst . data_level ;
2006-03-17 12:42:04 +03:00
state - > last_entry_offset = 0 ;
state - > flags = search - > t2ffirst . in . flags ;
2003-08-13 05:53:07 +04:00
/* setup for just a header in the reply */
2006-06-20 09:06:10 +04:00
TRANS2_CHECK ( trans2_setup_reply ( trans , 10 , 0 , 0 ) ) ;
2003-08-13 05:53:07 +04:00
2006-03-17 12:42:04 +03:00
op - > op_info = state ;
op - > send_fn = trans2_findfirst_send ;
2006-03-18 14:10:21 +03:00
return ntvfs_search_first ( req - > ntvfs , search , state , find_callback ) ;
2006-03-17 12:42:04 +03:00
}
/*
trans2 findnext send
*/
static NTSTATUS trans2_findnext_send ( struct trans_op * op )
{
struct smbsrv_request * req = op - > req ;
struct smb_trans2 * trans = op - > trans ;
union smb_search_next * search ;
struct find_state * state ;
uint8_t * param ;
2006-03-18 14:10:21 +03:00
TRANS2_CHECK_ASYNC_STATUS ( state , struct find_state ) ;
2006-03-17 12:42:04 +03:00
search = talloc_get_type ( state - > search , union smb_search_next ) ;
2003-08-13 05:53:07 +04:00
/* fill in the findfirst reply header */
param = trans - > out . params . data ;
2006-03-17 12:42:04 +03:00
SSVAL ( param , VWV ( 0 ) , search - > t2fnext . out . count ) ;
SSVAL ( param , VWV ( 1 ) , search - > t2fnext . out . end_of_search ) ;
SSVAL ( param , VWV ( 2 ) , 0 ) ;
SSVAL ( param , VWV ( 3 ) , state - > last_entry_offset ) ;
2003-08-13 05:53:07 +04:00
return NT_STATUS_OK ;
}
/*
trans2 findnext implementation
*/
2006-03-17 12:42:04 +03:00
static NTSTATUS trans2_findnext ( struct smbsrv_request * req , struct trans_op * op )
2003-08-13 05:53:07 +04:00
{
2006-03-17 12:42:04 +03:00
struct smb_trans2 * trans = op - > trans ;
union smb_search_next * search ;
2004-05-25 21:24:24 +04:00
uint16_t level ;
2006-03-17 12:42:04 +03:00
struct find_state * state ;
2003-08-13 05:53:07 +04:00
/* make sure we got all the parameters */
if ( trans - > in . params . length < 12 ) {
return NT_STATUS_FOOBAR ;
}
2006-03-17 12:42:04 +03:00
search = talloc ( op , union smb_search_next ) ;
NT_STATUS_HAVE_NO_MEMORY ( search ) ;
search - > t2fnext . in . handle = SVAL ( trans - > in . params . data , 0 ) ;
search - > t2fnext . in . max_count = SVAL ( trans - > in . params . data , 2 ) ;
level = SVAL ( trans - > in . params . data , 4 ) ;
search - > t2fnext . in . resume_key = IVAL ( trans - > in . params . data , 6 ) ;
search - > t2fnext . in . flags = SVAL ( trans - > in . params . data , 10 ) ;
2003-08-13 05:53:07 +04:00
2008-02-14 02:12:33 +03:00
smbsrv_blob_pull_string ( & req - > in . bufinfo , & trans - > in . params , 12 , & search - > t2fnext . in . last_name , 0 ) ;
2006-03-17 12:42:04 +03:00
if ( search - > t2fnext . in . last_name = = NULL ) {
2003-08-13 05:53:07 +04:00
return NT_STATUS_FOOBAR ;
}
2006-07-06 12:00:24 +04:00
search - > t2fnext . level = RAW_SEARCH_TRANS2 ;
search - > t2fnext . data_level = ( enum smb_search_data_level ) level ;
if ( search - > t2fnext . data_level > = RAW_SEARCH_DATA_GENERIC ) {
2003-08-13 05:53:07 +04:00
return NT_STATUS_INVALID_LEVEL ;
}
2006-07-06 12:00:24 +04:00
if ( search - > t2fnext . data_level = = RAW_SEARCH_DATA_EA_LIST ) {
2006-06-20 09:06:10 +04:00
TRANS2_CHECK ( ea_pull_name_list ( & trans - > in . data , req ,
2010-05-01 22:33:20 +04:00
& search - > t2fnext . in . num_names ,
2006-06-20 09:06:10 +04:00
& search - > t2fnext . in . ea_names ) ) ;
2004-12-18 07:38:43 +03:00
}
2003-08-13 05:53:07 +04:00
/* setup the private state structure that the backend will give us in the callback */
2006-03-17 12:42:04 +03:00
state = talloc ( op , struct find_state ) ;
NT_STATUS_HAVE_NO_MEMORY ( state ) ;
state - > op = op ;
state - > search = search ;
2006-07-06 12:00:24 +04:00
state - > data_level = search - > t2fnext . data_level ;
2006-03-17 12:42:04 +03:00
state - > last_entry_offset = 0 ;
state - > flags = search - > t2fnext . in . flags ;
2003-08-13 05:53:07 +04:00
/* setup for just a header in the reply */
2006-06-20 09:06:10 +04:00
TRANS2_CHECK ( trans2_setup_reply ( trans , 8 , 0 , 0 ) ) ;
2003-08-13 05:53:07 +04:00
2006-03-17 12:42:04 +03:00
op - > op_info = state ;
op - > send_fn = trans2_findnext_send ;
2003-08-13 05:53:07 +04:00
2006-03-18 14:10:21 +03:00
return ntvfs_search_next ( req - > ntvfs , search , state , find_callback ) ;
2003-08-13 05:53:07 +04:00
}
/*
backend for trans2 requests
*/
2006-03-17 12:42:04 +03:00
static NTSTATUS trans2_backend ( struct smbsrv_request * req , struct trans_op * op )
2003-08-13 05:53:07 +04:00
{
2006-03-17 12:42:04 +03:00
struct smb_trans2 * trans = op - > trans ;
2004-09-29 17:17:09 +04:00
NTSTATUS status ;
/* direct trans2 pass thru */
2006-03-18 14:10:21 +03:00
status = ntvfs_trans2 ( req - > ntvfs , trans ) ;
2004-09-29 17:17:09 +04:00
if ( ! NT_STATUS_EQUAL ( NT_STATUS_NOT_IMPLEMENTED , status ) ) {
return status ;
2003-08-13 05:53:07 +04:00
}
/* must have at least one setup word */
if ( trans - > in . setup_count < 1 ) {
return NT_STATUS_FOOBAR ;
}
/* the trans2 command is in setup[0] */
switch ( trans - > in . setup [ 0 ] ) {
2010-05-01 22:27:31 +04:00
case TRANSACT2_GET_DFS_REFERRAL :
return trans2_getdfsreferral ( req , op ) ;
2003-08-13 05:53:07 +04:00
case TRANSACT2_FINDFIRST :
2006-03-17 12:42:04 +03:00
return trans2_findfirst ( req , op ) ;
2003-08-13 05:53:07 +04:00
case TRANSACT2_FINDNEXT :
2006-03-17 12:42:04 +03:00
return trans2_findnext ( req , op ) ;
2003-08-13 05:53:07 +04:00
case TRANSACT2_QPATHINFO :
2006-03-17 12:42:04 +03:00
return trans2_qpathinfo ( req , op ) ;
2003-08-13 05:53:07 +04:00
case TRANSACT2_QFILEINFO :
2006-03-17 12:42:04 +03:00
return trans2_qfileinfo ( req , op ) ;
2003-08-13 05:53:07 +04:00
case TRANSACT2_SETFILEINFO :
2006-03-17 12:42:04 +03:00
return trans2_setfileinfo ( req , op ) ;
2003-08-13 05:53:07 +04:00
case TRANSACT2_SETPATHINFO :
2006-03-17 12:42:04 +03:00
return trans2_setpathinfo ( req , op ) ;
2003-08-13 05:53:07 +04:00
case TRANSACT2_QFSINFO :
2006-03-17 12:42:04 +03:00
return trans2_qfsinfo ( req , op ) ;
2004-11-06 10:58:45 +03:00
case TRANSACT2_OPEN :
2006-03-17 12:42:04 +03:00
return trans2_open ( req , op ) ;
2004-12-14 02:57:59 +03:00
case TRANSACT2_MKDIR :
2006-03-17 12:42:04 +03:00
return trans2_mkdir ( req , op ) ;
2003-08-13 05:53:07 +04:00
}
/* an unknown trans2 command */
return NT_STATUS_FOOBAR ;
}
2008-07-07 21:37:14 +04:00
int smbsrv_trans_partial_destructor ( struct smbsrv_trans_partial * tp )
2008-07-07 14:56:26 +04:00
{
DLIST_REMOVE ( tp - > req - > smb_conn - > trans_partial , tp ) ;
return 0 ;
}
2004-12-16 15:31:34 +03:00
/*
send a continue request
*/
2006-03-17 12:42:04 +03:00
static void reply_trans_continue ( struct smbsrv_request * req , uint8_t command ,
2004-12-16 15:31:34 +03:00
struct smb_trans2 * trans )
2003-08-13 05:53:07 +04:00
{
2008-07-07 14:56:26 +04:00
struct smbsrv_request * req2 ;
2004-12-16 15:31:34 +03:00
struct smbsrv_trans_partial * tp ;
int count ;
2003-08-13 05:53:07 +04:00
2004-12-16 15:31:34 +03:00
/* make sure they don't flood us */
for ( count = 0 , tp = req - > smb_conn - > trans_partial ; tp ; tp = tp - > next ) count + + ;
if ( count > 100 ) {
2006-03-06 19:19:27 +03:00
smbsrv_send_error ( req , NT_STATUS_INSUFFICIENT_RESOURCES ) ;
2003-08-13 05:53:07 +04:00
return ;
}
2005-01-27 10:08:20 +03:00
tp = talloc ( req , struct smbsrv_trans_partial ) ;
2003-08-13 05:53:07 +04:00
2008-07-07 14:56:26 +04:00
tp - > req = req ;
2008-07-07 21:37:14 +04:00
tp - > u . trans = trans ;
2004-12-16 15:31:34 +03:00
tp - > command = command ;
2003-08-13 05:53:07 +04:00
2004-12-16 15:31:34 +03:00
DLIST_ADD ( req - > smb_conn - > trans_partial , tp ) ;
2008-07-07 14:56:26 +04:00
talloc_set_destructor ( tp , smbsrv_trans_partial_destructor ) ;
req2 = smbsrv_setup_secondary_request ( req ) ;
2003-12-11 12:07:45 +03:00
2004-12-16 15:31:34 +03:00
/* send a 'please continue' reply */
2008-07-07 14:56:26 +04:00
smbsrv_setup_reply ( req2 , 0 , 0 ) ;
smbsrv_send_reply ( req2 ) ;
2004-12-16 15:31:34 +03:00
}
2003-08-13 05:53:07 +04:00
2004-12-16 15:31:34 +03:00
/*
answer a reconstructed trans request
*/
2006-03-18 14:10:21 +03:00
static void reply_trans_send ( struct ntvfs_request * ntvfs )
2004-12-16 15:31:34 +03:00
{
2006-03-18 14:10:21 +03:00
struct smbsrv_request * req ;
2006-03-17 12:42:04 +03:00
struct trans_op * op ;
struct smb_trans2 * trans ;
2004-12-16 15:31:34 +03:00
uint16_t params_left , data_left ;
uint8_t * params , * data ;
int i ;
2003-08-13 05:53:07 +04:00
2006-03-18 14:10:21 +03:00
SMBSRV_CHECK_ASYNC_STATUS_ERR ( op , struct trans_op ) ;
2006-03-17 12:42:04 +03:00
trans = op - > trans ;
/* if this function needs work to form the nttrans reply buffer, then
call that now */
if ( op - > send_fn ! = NULL ) {
NTSTATUS status ;
status = op - > send_fn ( op ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
smbsrv_send_error ( req , status ) ;
return ;
}
2003-08-13 05:53:07 +04:00
}
2004-12-16 15:31:34 +03:00
params_left = trans - > out . params . length ;
data_left = trans - > out . data . length ;
params = trans - > out . params . data ;
data = trans - > out . data . data ;
2004-10-29 07:48:49 +04:00
2006-03-06 19:19:27 +03:00
smbsrv_setup_reply ( req , 10 + trans - > out . setup_count , 0 ) ;
2004-10-29 07:48:49 +04:00
2006-03-18 14:10:21 +03:00
if ( ! NT_STATUS_IS_OK ( req - > ntvfs - > async_states - > status ) ) {
smbsrv_setup_error ( req , req - > ntvfs - > async_states - > status ) ;
2004-10-29 07:48:49 +04:00
}
2003-08-13 05:53:07 +04:00
/* we need to divide up the reply into chunks that fit into
the negotiated buffer size */
do {
2004-05-25 21:24:24 +04:00
uint16_t this_data , this_param , max_bytes ;
2010-01-05 20:42:54 +03:00
unsigned int align1 = 1 , align2 = ( params_left ? 2 : 0 ) ;
2004-10-29 07:48:49 +04:00
struct smbsrv_request * this_req ;
2003-08-13 05:53:07 +04:00
max_bytes = req_max_data ( req ) - ( align1 + align2 ) ;
this_param = params_left ;
if ( this_param > max_bytes ) {
this_param = max_bytes ;
}
max_bytes - = this_param ;
this_data = data_left ;
if ( this_data > max_bytes ) {
this_data = max_bytes ;
}
2004-10-29 07:48:49 +04:00
/* don't destroy unless this is the last chunk */
2010-05-01 22:33:20 +04:00
if ( params_left - this_param ! = 0 | |
2004-10-29 07:48:49 +04:00
data_left - this_data ! = 0 ) {
2006-03-06 19:19:27 +03:00
this_req = smbsrv_setup_secondary_request ( req ) ;
2004-10-29 07:48:49 +04:00
} else {
this_req = req ;
}
req_grow_data ( this_req , this_param + this_data + ( align1 + align2 ) ) ;
2003-08-13 05:53:07 +04:00
2004-12-16 15:31:34 +03:00
SSVAL ( this_req - > out . vwv , VWV ( 0 ) , trans - > out . params . length ) ;
SSVAL ( this_req - > out . vwv , VWV ( 1 ) , trans - > out . data . length ) ;
2004-10-29 07:48:49 +04:00
SSVAL ( this_req - > out . vwv , VWV ( 2 ) , 0 ) ;
2003-08-13 05:53:07 +04:00
2004-10-29 07:48:49 +04:00
SSVAL ( this_req - > out . vwv , VWV ( 3 ) , this_param ) ;
SSVAL ( this_req - > out . vwv , VWV ( 4 ) , align1 + PTR_DIFF ( this_req - > out . data , this_req - > out . hdr ) ) ;
2004-12-16 15:31:34 +03:00
SSVAL ( this_req - > out . vwv , VWV ( 5 ) , PTR_DIFF ( params , trans - > out . params . data ) ) ;
2003-08-13 05:53:07 +04:00
2004-10-29 07:48:49 +04:00
SSVAL ( this_req - > out . vwv , VWV ( 6 ) , this_data ) ;
2010-05-01 22:33:20 +04:00
SSVAL ( this_req - > out . vwv , VWV ( 7 ) , align1 + align2 +
2004-10-29 07:48:49 +04:00
PTR_DIFF ( this_req - > out . data + this_param , this_req - > out . hdr ) ) ;
2004-12-16 15:31:34 +03:00
SSVAL ( this_req - > out . vwv , VWV ( 8 ) , PTR_DIFF ( data , trans - > out . data . data ) ) ;
2003-08-13 05:53:07 +04:00
2008-07-07 16:00:53 +04:00
SCVAL ( this_req - > out . vwv , VWV ( 9 ) , trans - > out . setup_count ) ;
SCVAL ( this_req - > out . vwv , VWV ( 9 ) + 1 , 0 ) ; /* reserved */
2004-12-16 15:31:34 +03:00
for ( i = 0 ; i < trans - > out . setup_count ; i + + ) {
SSVAL ( this_req - > out . vwv , VWV ( 10 + i ) , trans - > out . setup [ i ] ) ;
2003-08-13 05:53:07 +04:00
}
2004-10-29 07:48:49 +04:00
memset ( this_req - > out . data , 0 , align1 ) ;
2003-08-13 05:53:07 +04:00
if ( this_param ! = 0 ) {
2004-10-29 07:48:49 +04:00
memcpy ( this_req - > out . data + align1 , params , this_param ) ;
2003-08-13 05:53:07 +04:00
}
2004-10-29 07:48:49 +04:00
memset ( this_req - > out . data + this_param + align1 , 0 , align2 ) ;
2003-08-13 05:53:07 +04:00
if ( this_data ! = 0 ) {
2004-10-29 07:48:49 +04:00
memcpy ( this_req - > out . data + this_param + align1 + align2 , data , this_data ) ;
2003-08-13 05:53:07 +04:00
}
params_left - = this_param ;
data_left - = this_data ;
params + = this_param ;
data + = this_data ;
2006-03-06 19:19:27 +03:00
smbsrv_send_reply ( this_req ) ;
2003-08-13 05:53:07 +04:00
} while ( params_left ! = 0 | | data_left ! = 0 ) ;
}
2003-12-11 12:07:45 +03:00
2006-03-17 12:42:04 +03:00
/*
answer a reconstructed trans request
*/
static void reply_trans_complete ( struct smbsrv_request * req , uint8_t command ,
struct smb_trans2 * trans )
{
struct trans_op * op ;
2006-03-18 14:10:21 +03:00
SMBSRV_TALLOC_IO_PTR ( op , struct trans_op ) ;
SMBSRV_SETUP_NTVFS_REQUEST ( reply_trans_send , NTVFS_ASYNC_STATE_MAY_ASYNC ) ;
2006-03-17 12:42:04 +03:00
op - > req = req ;
op - > trans = trans ;
op - > command = command ;
op - > op_info = NULL ;
op - > send_fn = NULL ;
/* its a full request, give it to the backend */
if ( command = = SMBtrans ) {
2006-03-18 14:10:21 +03:00
SMBSRV_CALL_NTVFS_BACKEND ( ntvfs_trans ( req - > ntvfs , trans ) ) ;
return ;
2006-03-17 12:42:04 +03:00
} else {
2006-03-18 14:10:21 +03:00
SMBSRV_CALL_NTVFS_BACKEND ( trans2_backend ( req , op ) ) ;
return ;
2006-03-17 12:42:04 +03:00
}
}
2004-12-16 15:31:34 +03:00
/*
Reply to an SMBtrans or SMBtrans2 request
*/
2006-03-06 18:04:59 +03:00
static void reply_trans_generic ( struct smbsrv_request * req , uint8_t command )
2004-12-16 15:31:34 +03:00
{
struct smb_trans2 * trans ;
int i ;
uint16_t param_ofs , data_ofs ;
uint16_t param_count , data_count ;
uint16_t param_total , data_total ;
/* parse request */
if ( req - > in . wct < 14 ) {
2006-03-06 19:19:27 +03:00
smbsrv_send_error ( req , NT_STATUS_INVALID_PARAMETER ) ;
2004-12-16 15:31:34 +03:00
return ;
}
2006-03-17 12:42:04 +03:00
trans = talloc ( req , struct smb_trans2 ) ;
if ( trans = = NULL ) {
smbsrv_send_error ( req , NT_STATUS_NO_MEMORY ) ;
return ;
}
2004-12-16 15:31:34 +03:00
param_total = SVAL ( req - > in . vwv , VWV ( 0 ) ) ;
data_total = SVAL ( req - > in . vwv , VWV ( 1 ) ) ;
trans - > in . max_param = SVAL ( req - > in . vwv , VWV ( 2 ) ) ;
trans - > in . max_data = SVAL ( req - > in . vwv , VWV ( 3 ) ) ;
trans - > in . max_setup = CVAL ( req - > in . vwv , VWV ( 4 ) ) ;
trans - > in . flags = SVAL ( req - > in . vwv , VWV ( 5 ) ) ;
trans - > in . timeout = IVAL ( req - > in . vwv , VWV ( 6 ) ) ;
param_count = SVAL ( req - > in . vwv , VWV ( 9 ) ) ;
param_ofs = SVAL ( req - > in . vwv , VWV ( 10 ) ) ;
data_count = SVAL ( req - > in . vwv , VWV ( 11 ) ) ;
data_ofs = SVAL ( req - > in . vwv , VWV ( 12 ) ) ;
trans - > in . setup_count = CVAL ( req - > in . vwv , VWV ( 13 ) ) ;
if ( req - > in . wct ! = 14 + trans - > in . setup_count ) {
2006-03-06 21:29:48 +03:00
smbsrv_send_error ( req , NT_STATUS_DOS ( ERRSRV , ERRerror ) ) ;
2004-12-16 15:31:34 +03:00
return ;
}
/* parse out the setup words */
2006-03-17 12:42:04 +03:00
trans - > in . setup = talloc_array ( trans , uint16_t , trans - > in . setup_count ) ;
2004-12-16 15:31:34 +03:00
if ( trans - > in . setup_count & & ! trans - > in . setup ) {
2006-03-06 19:19:27 +03:00
smbsrv_send_error ( req , NT_STATUS_NO_MEMORY ) ;
2004-12-16 15:31:34 +03:00
return ;
}
for ( i = 0 ; i < trans - > in . setup_count ; i + + ) {
trans - > in . setup [ i ] = SVAL ( req - > in . vwv , VWV ( 14 + i ) ) ;
}
if ( command = = SMBtrans ) {
2008-02-14 02:12:33 +03:00
req_pull_string ( & req - > in . bufinfo , & trans - > in . trans_name , req - > in . data , - 1 , STR_TERMINATE ) ;
2004-12-16 15:31:34 +03:00
}
2008-02-14 02:12:33 +03:00
if ( ! req_pull_blob ( & req - > in . bufinfo , req - > in . hdr + param_ofs , param_count , & trans - > in . params ) | |
! req_pull_blob ( & req - > in . bufinfo , req - > in . hdr + data_ofs , data_count , & trans - > in . data ) ) {
2006-03-06 19:19:27 +03:00
smbsrv_send_error ( req , NT_STATUS_FOOBAR ) ;
2004-12-16 15:31:34 +03:00
return ;
}
/* is it a partial request? if so, then send a 'send more' message */
if ( param_total > param_count | | data_total > data_count ) {
reply_trans_continue ( req , command , trans ) ;
return ;
}
reply_trans_complete ( req , command , trans ) ;
}
/*
Reply to an SMBtranss2 request
*/
static void reply_transs_generic ( struct smbsrv_request * req , uint8_t command )
{
struct smbsrv_trans_partial * tp ;
struct smb_trans2 * trans = NULL ;
uint16_t param_ofs , data_ofs ;
uint16_t param_count , data_count ;
uint16_t param_disp , data_disp ;
uint16_t param_total , data_total ;
DATA_BLOB params , data ;
2008-07-07 15:00:24 +04:00
uint8_t wct ;
if ( command = = SMBtrans2 ) {
wct = 9 ;
} else {
wct = 8 ;
}
2004-12-16 15:31:34 +03:00
2006-03-17 12:42:04 +03:00
/* parse request */
2008-07-07 15:00:24 +04:00
if ( req - > in . wct ! = wct ) {
/*
* TODO : add some error code tests
* w2k3 returns NT_STATUS_DOS ( ERRSRV , ERRerror ) here
*/
2006-03-17 12:42:04 +03:00
smbsrv_send_error ( req , NT_STATUS_INVALID_PARAMETER ) ;
return ;
}
2004-12-16 15:31:34 +03:00
for ( tp = req - > smb_conn - > trans_partial ; tp ; tp = tp - > next ) {
if ( tp - > command = = command & &
SVAL ( tp - > req - > in . hdr , HDR_MID ) = = SVAL ( req - > in . hdr , HDR_MID ) ) {
2006-03-17 12:42:04 +03:00
/* TODO: check the VUID, PID and TID too? */
2004-12-16 15:31:34 +03:00
break ;
}
}
if ( tp = = NULL ) {
2006-03-06 19:19:27 +03:00
smbsrv_send_error ( req , NT_STATUS_INVALID_PARAMETER ) ;
2004-12-16 15:31:34 +03:00
return ;
}
2008-07-07 21:37:14 +04:00
trans = tp - > u . trans ;
2004-12-16 15:31:34 +03:00
param_total = SVAL ( req - > in . vwv , VWV ( 0 ) ) ;
data_total = SVAL ( req - > in . vwv , VWV ( 1 ) ) ;
param_count = SVAL ( req - > in . vwv , VWV ( 2 ) ) ;
param_ofs = SVAL ( req - > in . vwv , VWV ( 3 ) ) ;
param_disp = SVAL ( req - > in . vwv , VWV ( 4 ) ) ;
data_count = SVAL ( req - > in . vwv , VWV ( 5 ) ) ;
data_ofs = SVAL ( req - > in . vwv , VWV ( 6 ) ) ;
data_disp = SVAL ( req - > in . vwv , VWV ( 7 ) ) ;
2008-02-14 02:12:33 +03:00
if ( ! req_pull_blob ( & req - > in . bufinfo , req - > in . hdr + param_ofs , param_count , & params ) | |
! req_pull_blob ( & req - > in . bufinfo , req - > in . hdr + data_ofs , data_count , & data ) ) {
2006-03-06 19:19:27 +03:00
smbsrv_send_error ( req , NT_STATUS_INVALID_PARAMETER ) ;
2004-12-16 15:31:34 +03:00
return ;
}
/* only allow contiguous requests */
if ( ( param_count ! = 0 & &
param_disp ! = trans - > in . params . length ) | |
2010-05-01 22:33:20 +04:00
( data_count ! = 0 & &
2004-12-16 15:31:34 +03:00
data_disp ! = trans - > in . data . length ) ) {
2006-03-06 19:19:27 +03:00
smbsrv_send_error ( req , NT_STATUS_INVALID_PARAMETER ) ;
2004-12-16 15:31:34 +03:00
return ;
}
/* add to the existing request */
if ( param_count ! = 0 ) {
2010-05-01 22:33:20 +04:00
trans - > in . params . data = talloc_realloc ( trans ,
trans - > in . params . data ,
uint8_t ,
2004-12-16 15:31:34 +03:00
param_disp + param_count ) ;
if ( trans - > in . params . data = = NULL ) {
2008-07-07 14:56:26 +04:00
smbsrv_send_error ( tp - > req , NT_STATUS_NO_MEMORY ) ;
return ;
2004-12-16 15:31:34 +03:00
}
trans - > in . params . length = param_disp + param_count ;
}
if ( data_count ! = 0 ) {
2010-05-01 22:33:20 +04:00
trans - > in . data . data = talloc_realloc ( trans ,
trans - > in . data . data ,
uint8_t ,
2004-12-16 15:31:34 +03:00
data_disp + data_count ) ;
if ( trans - > in . data . data = = NULL ) {
2008-07-07 14:56:26 +04:00
smbsrv_send_error ( tp - > req , NT_STATUS_NO_MEMORY ) ;
return ;
2004-12-16 15:31:34 +03:00
}
trans - > in . data . length = data_disp + data_count ;
}
memcpy ( trans - > in . params . data + param_disp , params . data , params . length ) ;
memcpy ( trans - > in . data . data + data_disp , data . data , data . length ) ;
/* the sequence number of the reply is taken from the last secondary
response */
tp - > req - > seq_num = req - > seq_num ;
/* we don't reply to Transs2 requests */
talloc_free ( req ) ;
if ( trans - > in . params . length = = param_total & &
trans - > in . data . length = = data_total ) {
/* its now complete */
2008-07-07 14:56:26 +04:00
req = tp - > req ;
talloc_free ( tp ) ;
reply_trans_complete ( req , command , trans ) ;
2004-12-16 15:31:34 +03:00
}
return ;
}
/*
Reply to an SMBtrans2
*/
2006-03-06 18:31:01 +03:00
void smbsrv_reply_trans2 ( struct smbsrv_request * req )
2003-12-11 12:07:45 +03:00
{
reply_trans_generic ( req , SMBtrans2 ) ;
}
2004-12-16 15:31:34 +03:00
/*
Reply to an SMBtrans
*/
2006-03-06 18:31:01 +03:00
void smbsrv_reply_trans ( struct smbsrv_request * req )
2003-12-11 12:07:45 +03:00
{
reply_trans_generic ( req , SMBtrans ) ;
}
2004-12-16 15:31:34 +03:00
/*
Reply to an SMBtranss request
*/
2006-03-06 18:31:01 +03:00
void smbsrv_reply_transs ( struct smbsrv_request * req )
2003-12-11 12:07:45 +03:00
{
2004-12-16 15:31:34 +03:00
reply_transs_generic ( req , SMBtrans ) ;
2003-12-11 12:07:45 +03:00
}
2004-12-16 15:31:34 +03:00
/*
Reply to an SMBtranss2 request
*/
2006-03-06 18:31:01 +03:00
void smbsrv_reply_transs2 ( struct smbsrv_request * req )
2004-12-16 15:31:34 +03:00
{
reply_transs_generic ( req , SMBtrans2 ) ;
}