2003-08-13 01:53:07 +00:00
/*
Unix SMB / CIFS implementation .
Main SMB reply routines
Copyright ( C ) Andrew Tridgell 1992 - 2003
2003-12-04 02:03:06 +00:00
Copyright ( C ) James J Myers 2003 < myersjj @ samba . org >
2003-08-13 01:53:07 +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
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program ; if not , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
/*
This file handles most of the reply_ calls that the server
makes to handle specific protocols
*/
# include "includes.h"
2004-11-02 07:18:24 +00:00
# include "smb_server/smb_server.h"
2005-12-27 22:51:30 +00:00
# include "ntvfs/ntvfs.h"
2004-11-02 07:18:24 +00:00
2003-08-13 01:53:07 +00:00
/* useful way of catching wct errors with file and line number */
# define REQ_CHECK_WCT(req, wcount) do { \
if ( ( req ) - > in . wct ! = ( wcount ) ) { \
DEBUG ( 1 , ( " Unexpected WCT %d at %s(%d) - expected %d \n " , \
( req ) - > in . wct , __FILE__ , __LINE__ , wcount ) ) ; \
req_reply_dos_error ( req , ERRSRV , ERRerror ) ; \
return ; \
} } while ( 0 )
2004-10-28 21:48:53 +00:00
/* check req->async_states->status and if not OK then send an error reply */
2003-08-13 01:53:07 +00:00
# define CHECK_ASYNC_STATUS do { \
2004-10-28 21:48:53 +00:00
if ( ! NT_STATUS_IS_OK ( req - > async_states - > status ) ) { \
req_reply_error ( req , req - > async_states - > status ) ; \
2003-08-13 01:53:07 +00:00
return ; \
} } while ( 0 )
/* useful wrapper for talloc with NO_MEMORY reply */
# define REQ_TALLOC(ptr, size) do { \
2005-01-06 03:06:58 +00:00
ptr = talloc_size ( req , size ) ; \
2003-08-13 01:53:07 +00:00
if ( ! ptr ) { \
req_reply_error ( req , NT_STATUS_NO_MEMORY ) ; \
return ; \
} } while ( 0 )
/*
check if the backend wants to handle the request asynchronously .
if it wants it handled synchronously then call the send function
immediately
*/
# define REQ_ASYNC_TAIL do { \
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_ASYNC ) ) { \
req - > async_states - > send_fn ( req ) ; \
2003-08-13 01:53:07 +00:00
} } while ( 0 )
/* zero out some reserved fields in a reply */
# define REQ_VWV_RESERVED(start, count) memset(req->out.vwv + VWV(start), 0, (count)*2)
/****************************************************************************
Reply to a simple request ( async send )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_simple_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
CHECK_ASYNC_STATUS ;
req_setup_reply ( req , 0 , 0 ) ;
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a tcon .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_tcon ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_tcon con ;
NTSTATUS status ;
2004-12-04 13:56:25 +00:00
uint8_t * p ;
2003-08-13 01:53:07 +00:00
/* parse request */
REQ_CHECK_WCT ( req , 0 ) ;
con . tcon . level = RAW_TCON_TCON ;
p = req - > in . data ;
p + = req_pull_ascii4 ( req , & con . tcon . in . service , p , STR_TERMINATE ) ;
p + = req_pull_ascii4 ( req , & con . tcon . in . password , p , STR_TERMINATE ) ;
p + = req_pull_ascii4 ( req , & con . tcon . in . dev , p , STR_TERMINATE ) ;
if ( ! con . tcon . in . service | | ! con . tcon . in . password | | ! con . tcon . in . dev ) {
req_reply_error ( req , NT_STATUS_INVALID_PARAMETER ) ;
return ;
}
/* call backend */
status = tcon_backend ( req , & con ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
req_reply_error ( req , status ) ;
return ;
}
/* construct reply */
req_setup_reply ( req , 2 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , con . tcon . out . max_xmit ) ;
2005-01-13 18:49:10 +00:00
SSVAL ( req - > out . vwv , VWV ( 1 ) , con . tcon . out . tid ) ;
SSVAL ( req - > out . hdr , HDR_TID , req - > tcon - > tid ) ;
2003-08-13 01:53:07 +00:00
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a tcon and X .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_tcon_and_X ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
NTSTATUS status ;
union smb_tcon con ;
2004-12-04 13:56:25 +00:00
uint8_t * p ;
2004-05-25 17:24:24 +00:00
uint16_t passlen ;
2003-08-13 01:53:07 +00:00
con . tconx . level = RAW_TCON_TCONX ;
/* parse request */
REQ_CHECK_WCT ( req , 4 ) ;
con . tconx . in . flags = SVAL ( req - > in . vwv , VWV ( 2 ) ) ;
passlen = SVAL ( req - > in . vwv , VWV ( 3 ) ) ;
p = req - > in . data ;
if ( ! req_pull_blob ( req , p , passlen , & con . tconx . in . password ) ) {
req_reply_error ( req , NT_STATUS_ILL_FORMED_PASSWORD ) ;
return ;
}
p + = passlen ;
p + = req_pull_string ( req , & con . tconx . in . path , p , - 1 , STR_TERMINATE ) ;
p + = req_pull_string ( req , & con . tconx . in . device , p , - 1 , STR_ASCII ) ;
if ( ! con . tconx . in . path | | ! con . tconx . in . device ) {
req_reply_error ( req , NT_STATUS_BAD_DEVICE_TYPE ) ;
return ;
}
/* call backend */
status = tcon_backend ( req , & con ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
req_reply_error ( req , status ) ;
return ;
}
2004-06-27 11:06:10 +00:00
/* construct reply - two variants */
2004-06-29 07:40:14 +00:00
if ( req - > smb_conn - > negotiate . protocol < PROTOCOL_NT1 ) {
2003-08-13 01:53:07 +00:00
req_setup_reply ( req , 2 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , SMB_CHAIN_NONE ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , 0 ) ;
req_push_str ( req , NULL , con . tconx . out . dev_type , - 1 , STR_TERMINATE | STR_ASCII ) ;
} else {
req_setup_reply ( req , 3 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , SMB_CHAIN_NONE ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 2 ) , con . tconx . out . options ) ;
req_push_str ( req , NULL , con . tconx . out . dev_type , - 1 , STR_TERMINATE | STR_ASCII ) ;
req_push_str ( req , NULL , con . tconx . out . fs_type , - 1 , STR_TERMINATE ) ;
}
/* set the incoming and outgoing tid to the just created one */
2005-01-13 18:49:10 +00:00
SSVAL ( req - > in . hdr , HDR_TID , con . tconx . out . tid ) ;
SSVAL ( req - > out . hdr , HDR_TID , con . tconx . out . tid ) ;
2003-08-13 01:53:07 +00:00
2006-03-06 15:00:42 +00:00
smbsrv_chain_reply ( req ) ;
2003-08-13 01:53:07 +00:00
}
/****************************************************************************
Reply to an unknown request
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_unknown ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
int type ;
type = CVAL ( req - > in . hdr , HDR_COM ) ;
DEBUG ( 0 , ( " unknown command type %d (0x%X) \n " , type , type ) ) ;
req_reply_dos_error ( req , ERRSRV , ERRunknownsmb ) ;
}
/****************************************************************************
Reply to an ioctl ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_ioctl_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_ioctl * io = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* the +1 is for nicer alignment */
2003-12-04 02:03:06 +00:00
req_setup_reply ( req , 8 , io - > ioctl . out . blob . length + 1 ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , io - > ioctl . out . blob . length ) ;
SSVAL ( req - > out . vwv , VWV ( 5 ) , io - > ioctl . out . blob . length ) ;
2003-08-13 01:53:07 +00:00
SSVAL ( req - > out . vwv , VWV ( 6 ) , PTR_DIFF ( req - > out . data , req - > out . hdr ) + 1 ) ;
2003-12-04 02:03:06 +00:00
memcpy ( req - > out . data + 1 , io - > ioctl . out . blob . data , io - > ioctl . out . blob . length ) ;
2003-08-13 01:53:07 +00:00
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to an ioctl .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_ioctl ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2003-12-04 02:03:06 +00:00
union smb_ioctl * io ;
2003-08-13 01:53:07 +00:00
This patch adds a better dcerpc server infastructure.
1.) We now register endpoint servers add startup via register_backend()
and later use the smb.conf 'dcerpc endpoint servers' parameter to setup the dcesrv_context
2.) each endpoint server can register at context creation time as much interfaces as it wants
(multiple interfaces on one endpoint are supported!)
(NOTE: there's a difference between 'endpoint server' and 'endpoint'!
for details look at rpc_server/dcesrv_server.h)
3.) one endpoint can have a security descriptor registered to it self
this will be checked in the future when a client wants to connect
to an smb pipe endpoint.
4.) we now have a 'remote' endpoint server, which works like the ntvfs_cifs module
it takes this options in the [globals] section:
dcerpc remote:interfaces = srvsvc, winreg, w32time, epmapper
dcerpc remote:binding = ...
dcerpc remote:user = ...
dcerpc remote:password = ...
5.) we currently have tree endpoint servers: epmapper, rpcecho and remote
the default for the 'dcerpc endpiont servers = epmapper, rpcecho'
for testing you can also do
dcerpc endpoint servers = rpcecho, remote, epmapper
dcerpc remote:interfaces = srvsvc, samr, netlogon
6,) please notice the the epmapper now only returns NO_ENTRIES
(but I think we'll find a solution for this too:-)
7.) also there're some other stuff left, but step by step :-)
This patch also includes updates for the
register_subsystem() , ntvfs_init(), and some other funtions
to check for duplicate subsystem registration
metze
(hmmm, my first large commit...I hope it works as supposed :-)
(This used to be commit 917e45dafd5be4c2cd90ff425b8d6f8403122349)
2004-01-08 22:55:27 +00:00
/* parse request */
2003-08-13 01:53:07 +00:00
REQ_CHECK_WCT ( req , 3 ) ;
REQ_TALLOC ( io , sizeof ( * io ) ) ;
2003-12-04 02:03:06 +00:00
io - > ioctl . level = RAW_IOCTL_IOCTL ;
io - > ioctl . in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 0 ) ) ;
io - > ioctl . in . request = IVAL ( req - > in . vwv , VWV ( 1 ) ) ;
2003-08-13 01:53:07 +00:00
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_ioctl_send ;
req - > async_states - > private_data = io ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_ioctl ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a chkpth .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_chkpth ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
struct smb_chkpath * io ;
REQ_TALLOC ( io , sizeof ( * io ) ) ;
req_pull_ascii4 ( req , & io - > in . path , req - > in . data , STR_TERMINATE ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_simple_send ;
2003-08-13 01:53:07 +00:00
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_chkpath ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a getatr ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_getatr_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_fileinfo * st = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* construct reply */
req_setup_reply ( req , 10 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , st - > getattr . out . attrib ) ;
2004-06-29 07:40:14 +00:00
srv_push_dos_date3 ( req - > smb_conn , req - > out . vwv , VWV ( 1 ) , st - > getattr . out . write_time ) ;
2003-08-13 01:53:07 +00:00
SIVAL ( req - > out . vwv , VWV ( 3 ) , st - > getattr . out . size ) ;
REQ_VWV_RESERVED ( 5 , 5 ) ;
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a getatr .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_getatr ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_fileinfo * st ;
REQ_TALLOC ( st , sizeof ( * st ) ) ;
st - > getattr . level = RAW_FILEINFO_GETATTR ;
/* parse request */
req_pull_ascii4 ( req , & st - > getattr . in . fname , req - > in . data , STR_TERMINATE ) ;
if ( ! st - > getattr . in . fname ) {
req_reply_error ( req , NT_STATUS_OBJECT_NAME_NOT_FOUND ) ;
return ;
}
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_getatr_send ;
req - > async_states - > private_data = st ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_qpathinfo ( req , st ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a setatr .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_setatr ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_setfileinfo * st ;
/* parse request */
REQ_CHECK_WCT ( req , 8 ) ;
REQ_TALLOC ( st , sizeof ( * st ) ) ;
st - > setattr . level = RAW_SFILEINFO_SETATTR ;
st - > setattr . in . attrib = SVAL ( req - > in . vwv , VWV ( 0 ) ) ;
2004-06-29 07:40:14 +00:00
st - > setattr . in . write_time = srv_pull_dos_date3 ( req - > smb_conn , req - > in . vwv + VWV ( 1 ) ) ;
2003-08-13 01:53:07 +00:00
req_pull_ascii4 ( req , & st - > setattr . file . fname , req - > in . data , STR_TERMINATE ) ;
if ( ! st - > setattr . file . fname ) {
req_reply_error ( req , NT_STATUS_OBJECT_NAME_NOT_FOUND ) ;
return ;
}
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_simple_send ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_setpathinfo ( req , st ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a dskattr ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_dskattr_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_fsinfo * fs = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* construct reply */
req_setup_reply ( req , 5 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , fs - > dskattr . out . units_total ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , fs - > dskattr . out . blocks_per_unit ) ;
SSVAL ( req - > out . vwv , VWV ( 2 ) , fs - > dskattr . out . block_size ) ;
SSVAL ( req - > out . vwv , VWV ( 3 ) , fs - > dskattr . out . units_free ) ;
REQ_VWV_RESERVED ( 4 , 1 ) ;
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a dskattr .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_dskattr ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_fsinfo * fs ;
REQ_TALLOC ( fs , sizeof ( * fs ) ) ;
fs - > dskattr . level = RAW_QFS_DSKATTR ;
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_dskattr_send ;
req - > async_states - > private_data = fs ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_fsinfo ( req , fs ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to an open ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_open_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_open * oi = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* construct reply */
req_setup_reply ( req , 7 , 0 ) ;
2004-11-02 04:17:30 +00:00
SSVAL ( req - > out . vwv , VWV ( 0 ) , oi - > openold . out . fnum ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , oi - > openold . out . attrib ) ;
srv_push_dos_date3 ( req - > smb_conn , req - > out . vwv , VWV ( 2 ) , oi - > openold . out . write_time ) ;
SIVAL ( req - > out . vwv , VWV ( 4 ) , oi - > openold . out . size ) ;
SSVAL ( req - > out . vwv , VWV ( 6 ) , oi - > openold . out . rmode ) ;
2003-08-13 01:53:07 +00:00
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to an open .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_open ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_open * oi ;
/* parse request */
REQ_CHECK_WCT ( req , 2 ) ;
REQ_TALLOC ( oi , sizeof ( * oi ) ) ;
2004-11-02 04:17:30 +00:00
oi - > openold . level = RAW_OPEN_OPEN ;
2004-12-13 10:48:21 +00:00
oi - > openold . in . open_mode = SVAL ( req - > in . vwv , VWV ( 0 ) ) ;
2004-11-02 04:17:30 +00:00
oi - > openold . in . search_attrs = SVAL ( req - > in . vwv , VWV ( 1 ) ) ;
2003-08-13 01:53:07 +00:00
2004-11-02 04:17:30 +00:00
req_pull_ascii4 ( req , & oi - > openold . in . fname , req - > in . data , STR_TERMINATE ) ;
2003-08-13 01:53:07 +00:00
2004-11-02 04:17:30 +00:00
if ( ! oi - > openold . in . fname ) {
2003-08-13 01:53:07 +00:00
req_reply_error ( req , NT_STATUS_OBJECT_NAME_NOT_FOUND ) ;
return ;
}
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_open_send ;
req - > async_states - > private_data = oi ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-11-02 04:51:57 +00:00
req - > async_states - > status = ntvfs_openfile ( req , oi ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to an open and X ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_open_and_X_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_open * oi = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* build the reply */
if ( oi - > openx . in . flags & OPENX_FLAGS_EXTENDED_RETURN ) {
req_setup_reply ( req , 19 , 0 ) ;
} else {
req_setup_reply ( req , 15 , 0 ) ;
}
SSVAL ( req - > out . vwv , VWV ( 0 ) , SMB_CHAIN_NONE ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 2 ) , oi - > openx . out . fnum ) ;
SSVAL ( req - > out . vwv , VWV ( 3 ) , oi - > openx . out . attrib ) ;
2004-06-29 07:40:14 +00:00
srv_push_dos_date3 ( req - > smb_conn , req - > out . vwv , VWV ( 4 ) , oi - > openx . out . write_time ) ;
2003-08-13 01:53:07 +00:00
SIVAL ( req - > out . vwv , VWV ( 6 ) , oi - > openx . out . size ) ;
SSVAL ( req - > out . vwv , VWV ( 8 ) , oi - > openx . out . access ) ;
SSVAL ( req - > out . vwv , VWV ( 9 ) , oi - > openx . out . ftype ) ;
SSVAL ( req - > out . vwv , VWV ( 10 ) , oi - > openx . out . devstate ) ;
SSVAL ( req - > out . vwv , VWV ( 11 ) , oi - > openx . out . action ) ;
SIVAL ( req - > out . vwv , VWV ( 12 ) , oi - > openx . out . unique_fid ) ;
SSVAL ( req - > out . vwv , VWV ( 14 ) , 0 ) ; /* reserved */
if ( oi - > openx . in . flags & OPENX_FLAGS_EXTENDED_RETURN ) {
SIVAL ( req - > out . vwv , VWV ( 15 ) , oi - > openx . out . access_mask ) ;
REQ_VWV_RESERVED ( 17 , 2 ) ;
}
2004-12-26 08:13:01 +00:00
req - > chained_fnum = oi - > openx . out . fnum ;
2006-03-06 15:00:42 +00:00
smbsrv_chain_reply ( req ) ;
2003-08-13 01:53:07 +00:00
}
/****************************************************************************
Reply to an open and X .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_open_and_X ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_open * oi ;
/* parse the request */
REQ_CHECK_WCT ( req , 15 ) ;
REQ_TALLOC ( oi , sizeof ( * oi ) ) ;
oi - > openx . level = RAW_OPEN_OPENX ;
oi - > openx . in . flags = SVAL ( req - > in . vwv , VWV ( 2 ) ) ;
oi - > openx . in . open_mode = SVAL ( req - > in . vwv , VWV ( 3 ) ) ;
oi - > openx . in . search_attrs = SVAL ( req - > in . vwv , VWV ( 4 ) ) ;
oi - > openx . in . file_attrs = SVAL ( req - > in . vwv , VWV ( 5 ) ) ;
2004-06-29 07:40:14 +00:00
oi - > openx . in . write_time = srv_pull_dos_date3 ( req - > smb_conn , req - > in . vwv + VWV ( 6 ) ) ;
2003-08-13 01:53:07 +00:00
oi - > openx . in . open_func = SVAL ( req - > in . vwv , VWV ( 8 ) ) ;
oi - > openx . in . size = IVAL ( req - > in . vwv , VWV ( 9 ) ) ;
oi - > openx . in . timeout = IVAL ( req - > in . vwv , VWV ( 11 ) ) ;
req_pull_ascii4 ( req , & oi - > openx . in . fname , req - > in . data , STR_TERMINATE ) ;
if ( ! oi - > openx . in . fname ) {
req_reply_error ( req , NT_STATUS_OBJECT_NAME_NOT_FOUND ) ;
return ;
}
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_open_and_X_send ;
req - > async_states - > private_data = oi ;
2003-08-13 01:53:07 +00:00
/* call the backend */
2004-11-02 04:51:57 +00:00
req - > async_states - > status = ntvfs_openfile ( req , oi ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a mknew or a create .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_mknew_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_open * oi = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* build the reply */
req_setup_reply ( req , 1 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , oi - > mknew . out . fnum ) ;
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a mknew or a create .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_mknew ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_open * oi ;
/* parse the request */
REQ_CHECK_WCT ( req , 3 ) ;
REQ_TALLOC ( oi , sizeof ( * oi ) ) ;
2004-11-04 13:15:22 +00:00
if ( CVAL ( req - > in . hdr , HDR_COM ) = = SMBmknew ) {
oi - > mknew . level = RAW_OPEN_MKNEW ;
} else {
oi - > mknew . level = RAW_OPEN_CREATE ;
}
2003-08-13 01:53:07 +00:00
oi - > mknew . in . attrib = SVAL ( req - > in . vwv , VWV ( 0 ) ) ;
2004-06-29 07:40:14 +00:00
oi - > mknew . in . write_time = srv_pull_dos_date3 ( req - > smb_conn , req - > in . vwv + VWV ( 1 ) ) ;
2003-08-13 01:53:07 +00:00
req_pull_ascii4 ( req , & oi - > mknew . in . fname , req - > in . data , STR_TERMINATE ) ;
if ( ! oi - > mknew . in . fname ) {
req_reply_error ( req , NT_STATUS_OBJECT_NAME_NOT_FOUND ) ;
return ;
}
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_mknew_send ;
req - > async_states - > private_data = oi ;
2003-08-13 01:53:07 +00:00
/* call the backend */
2004-11-02 04:51:57 +00:00
req - > async_states - > status = ntvfs_openfile ( req , oi ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a create temporary file ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_ctemp_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_open * oi = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* build the reply */
req_setup_reply ( req , 1 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , oi - > ctemp . out . fnum ) ;
/* the returned filename is relative to the directory */
2004-11-06 07:58:45 +00:00
req_push_str ( req , NULL , oi - > ctemp . out . name , - 1 , STR_TERMINATE | STR_ASCII ) ;
2003-08-13 01:53:07 +00:00
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a create temporary file .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_ctemp ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_open * oi ;
/* parse the request */
REQ_CHECK_WCT ( req , 3 ) ;
REQ_TALLOC ( oi , sizeof ( * oi ) ) ;
oi - > ctemp . level = RAW_OPEN_CTEMP ;
oi - > ctemp . in . attrib = SVAL ( req - > in . vwv , VWV ( 0 ) ) ;
2004-06-29 07:40:14 +00:00
oi - > ctemp . in . write_time = srv_pull_dos_date3 ( req - > smb_conn , req - > in . vwv + VWV ( 1 ) ) ;
2003-08-13 01:53:07 +00:00
/* the filename is actually a directory name, the server provides a filename
in that directory */
req_pull_ascii4 ( req , & oi - > ctemp . in . directory , req - > in . data , STR_TERMINATE ) ;
if ( ! oi - > ctemp . in . directory ) {
req_reply_error ( req , NT_STATUS_OBJECT_NAME_NOT_FOUND ) ;
return ;
}
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_ctemp_send ;
req - > async_states - > private_data = oi ;
2003-08-13 01:53:07 +00:00
/* call the backend */
2004-11-02 04:51:57 +00:00
req - > async_states - > status = ntvfs_openfile ( req , oi ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a unlink
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_unlink ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
struct smb_unlink * unl ;
/* parse the request */
REQ_CHECK_WCT ( req , 1 ) ;
REQ_TALLOC ( unl , sizeof ( * unl ) ) ;
unl - > in . attrib = SVAL ( req - > in . vwv , VWV ( 0 ) ) ;
req_pull_ascii4 ( req , & unl - > in . pattern , req - > in . data , STR_TERMINATE ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_simple_send ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_unlink ( req , unl ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a readbraw ( core + protocol ) .
this is a strange packet because it doesn ' t use a standard SMB header in the reply ,
only the 4 byte NBT header
This command must be replied to synchronously
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_readbraw ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
NTSTATUS status ;
union smb_read io ;
io . readbraw . level = RAW_READ_READBRAW ;
2004-06-27 11:06:10 +00:00
/* there are two variants, one with 10 and one with 8 command words */
r3081: several updates to ntvfs and server side async request handling in
preparation for the full share modes and ntcreatex code that I am
working on.
highlights include:
- changed the way a backend determines if it is allowed to process a
request asynchronously. The previous method of looking at the
send_fn caused problems when an intermediate ntvfs module disabled
it, and the caller then wanted to finished processing using this
function. The new method is a REQ_CONTROL_MAY_ASYNC flag in
req->control_flags, which is also a bit easier to read
- fixed 2 bugs in the readbraw server code. One related to trying to
answer a readbraw with smb signing (which can't work, and crashed
our signing code), the second related to error handling, which
attempted to send a normal SMB error packet, when readbraw must
send a 0 read reply (as it has no header)
- added several more ntvfs_generic.c generic mapping functions. This
means that backends no longer need to implement such esoteric
functions as SMBwriteunlock() if they don't want to. The backend
can just request the mapping layer turn it into a write followed by
an unlock. This makes the backends considerably simpler as they
only need to implement one style of each function for lock, read,
write, open etc, rather than the full host of functions that SMB
provides. A backend can still choose to implement them
individually, of course, and the CIFS backend does that.
- simplified the generic structures to make them identical to the
principal call for several common SMB calls (such as
RAW_WRITE_GENERIC now being an alias for RAW_WRITE_WRITEX).
- started rewriting the pvfs_open() code in preparation for the full
ntcreatex semantics.
- in pvfs_open and ipc_open, initially allocate the open file
structure as a child of the request, so on error we don't need to
clean up. Then when we are going to succeed the open steal the
pointer into the long term backend context. This makes for much
simpler error handling (and fixes some bugs)
- use a destructor in the ipc backend to make sure that everthing is
cleaned up on receive error conditions.
- switched the ipc backend to using idtree for fnum allocation
- in the ntvfs_generic mapping routines, use a allocated secondary
structure not a stack structure to ensure the request pointer
remains valid even if the backend replies async.
(This used to be commit 3457c1836c09c82956697eb21627dfa2ed37682e)
2004-10-20 08:28:31 +00:00
if ( req - > in . wct < 8 ) {
goto failed ;
2003-08-13 01:53:07 +00:00
}
io . readbraw . in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 0 ) ) ;
io . readbraw . in . offset = IVAL ( req - > in . vwv , VWV ( 1 ) ) ;
2004-10-20 11:11:55 +00:00
io . readbraw . in . maxcnt = SVAL ( req - > in . vwv , VWV ( 3 ) ) ;
io . readbraw . in . mincnt = SVAL ( req - > in . vwv , VWV ( 4 ) ) ;
2003-08-13 01:53:07 +00:00
io . readbraw . in . timeout = IVAL ( req - > in . vwv , VWV ( 5 ) ) ;
2004-06-27 11:06:10 +00:00
/* the 64 bit variant */
2003-08-13 01:53:07 +00:00
if ( req - > in . wct = = 10 ) {
2004-05-25 16:24:13 +00:00
uint32_t offset_high = IVAL ( req - > in . vwv , VWV ( 8 ) ) ;
2004-11-01 20:21:54 +00:00
io . readbraw . in . offset | = ( ( ( off_t ) offset_high ) < < 32 ) ;
2003-08-13 01:53:07 +00:00
}
/* before calling the backend we setup the raw buffer. This
* saves a copy later */
req - > out . size = io . readbraw . in . maxcnt + NBT_HDR_SIZE ;
2005-01-06 03:06:58 +00:00
req - > out . buffer = talloc_size ( req , req - > out . size ) ;
2003-08-13 01:53:07 +00:00
if ( req - > out . buffer = = NULL ) {
goto failed ;
}
2003-12-04 11:01:58 +00:00
SIVAL ( req - > out . buffer , 0 , 0 ) ; /* init NBT header */
2003-08-13 01:53:07 +00:00
/* tell the backend where to put the data */
io . readbraw . out . data = req - > out . buffer + NBT_HDR_SIZE ;
/* call the backend */
2004-09-29 13:17:09 +00:00
status = ntvfs_read ( req , & io ) ;
2003-08-13 01:53:07 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto failed ;
}
req - > out . size = io . readbraw . out . nread + NBT_HDR_SIZE ;
r3081: several updates to ntvfs and server side async request handling in
preparation for the full share modes and ntcreatex code that I am
working on.
highlights include:
- changed the way a backend determines if it is allowed to process a
request asynchronously. The previous method of looking at the
send_fn caused problems when an intermediate ntvfs module disabled
it, and the caller then wanted to finished processing using this
function. The new method is a REQ_CONTROL_MAY_ASYNC flag in
req->control_flags, which is also a bit easier to read
- fixed 2 bugs in the readbraw server code. One related to trying to
answer a readbraw with smb signing (which can't work, and crashed
our signing code), the second related to error handling, which
attempted to send a normal SMB error packet, when readbraw must
send a 0 read reply (as it has no header)
- added several more ntvfs_generic.c generic mapping functions. This
means that backends no longer need to implement such esoteric
functions as SMBwriteunlock() if they don't want to. The backend
can just request the mapping layer turn it into a write followed by
an unlock. This makes the backends considerably simpler as they
only need to implement one style of each function for lock, read,
write, open etc, rather than the full host of functions that SMB
provides. A backend can still choose to implement them
individually, of course, and the CIFS backend does that.
- simplified the generic structures to make them identical to the
principal call for several common SMB calls (such as
RAW_WRITE_GENERIC now being an alias for RAW_WRITE_WRITEX).
- started rewriting the pvfs_open() code in preparation for the full
ntcreatex semantics.
- in pvfs_open and ipc_open, initially allocate the open file
structure as a child of the request, so on error we don't need to
clean up. Then when we are going to succeed the open steal the
pointer into the long term backend context. This makes for much
simpler error handling (and fixes some bugs)
- use a destructor in the ipc backend to make sure that everthing is
cleaned up on receive error conditions.
- switched the ipc backend to using idtree for fnum allocation
- in the ntvfs_generic mapping routines, use a allocated secondary
structure not a stack structure to ensure the request pointer
remains valid even if the backend replies async.
(This used to be commit 3457c1836c09c82956697eb21627dfa2ed37682e)
2004-10-20 08:28:31 +00:00
req_send_reply_nosign ( req ) ;
2003-12-04 11:01:58 +00:00
return ;
2003-08-13 01:53:07 +00:00
failed :
/* any failure in readbraw is equivalent to reading zero bytes */
req - > out . size = 4 ;
2005-01-06 03:06:58 +00:00
req - > out . buffer = talloc_size ( req , req - > out . size ) ;
2003-12-04 11:01:58 +00:00
SIVAL ( req - > out . buffer , 0 , 0 ) ; /* init NBT header */
2004-05-22 11:16:21 +00:00
req_send_reply_nosign ( req ) ;
2003-08-13 01:53:07 +00:00
}
/****************************************************************************
Reply to a lockread ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_lockread_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_read * io = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* trim packet */
2003-12-04 11:01:58 +00:00
io - > lockread . out . nread = MIN ( io - > lockread . out . nread ,
req_max_data ( req ) - 3 ) ;
2003-08-13 01:53:07 +00:00
req_grow_data ( req , 3 + io - > lockread . out . nread ) ;
/* construct reply */
SSVAL ( req - > out . vwv , VWV ( 0 ) , io - > lockread . out . nread ) ;
REQ_VWV_RESERVED ( 1 , 4 ) ;
SCVAL ( req - > out . data , 0 , SMB_DATA_BLOCK ) ;
SSVAL ( req - > out . data , 1 , io - > lockread . out . nread ) ;
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a lockread ( core + protocol ) .
note that the lock is a write lock , not a read lock !
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_lockread ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_read * io ;
/* parse request */
REQ_CHECK_WCT ( req , 5 ) ;
REQ_TALLOC ( io , sizeof ( * io ) ) ;
io - > lockread . level = RAW_READ_LOCKREAD ;
io - > lockread . in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 0 ) ) ;
io - > lockread . in . count = SVAL ( req - > in . vwv , VWV ( 1 ) ) ;
io - > lockread . in . offset = IVAL ( req - > in . vwv , VWV ( 2 ) ) ;
io - > lockread . in . remaining = SVAL ( req - > in . vwv , VWV ( 4 ) ) ;
/* setup the reply packet assuming the maximum possible read */
req_setup_reply ( req , 5 , 3 + io - > lockread . in . count ) ;
/* tell the backend where to put the data */
io - > lockread . out . data = req - > out . data + 3 ;
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_lockread_send ;
req - > async_states - > private_data = io ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_read ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a read ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_read_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_read * io = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* trim packet */
2003-12-04 11:01:58 +00:00
io - > read . out . nread = MIN ( io - > read . out . nread ,
req_max_data ( req ) - 3 ) ;
2003-08-13 01:53:07 +00:00
req_grow_data ( req , 3 + io - > read . out . nread ) ;
/* construct reply */
SSVAL ( req - > out . vwv , VWV ( 0 ) , io - > read . out . nread ) ;
REQ_VWV_RESERVED ( 1 , 4 ) ;
SCVAL ( req - > out . data , 0 , SMB_DATA_BLOCK ) ;
SSVAL ( req - > out . data , 1 , io - > read . out . nread ) ;
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a read .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_read ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_read * io ;
/* parse request */
REQ_CHECK_WCT ( req , 5 ) ;
REQ_TALLOC ( io , sizeof ( * io ) ) ;
io - > read . level = RAW_READ_READ ;
io - > read . in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 0 ) ) ;
io - > read . in . count = SVAL ( req - > in . vwv , VWV ( 1 ) ) ;
io - > read . in . offset = IVAL ( req - > in . vwv , VWV ( 2 ) ) ;
io - > read . in . remaining = SVAL ( req - > in . vwv , VWV ( 4 ) ) ;
/* setup the reply packet assuming the maximum possible read */
req_setup_reply ( req , 5 , 3 + io - > read . in . count ) ;
/* tell the backend where to put the data */
2003-12-04 11:01:58 +00:00
io - > read . out . data = req - > out . data + 3 ;
2003-08-13 01:53:07 +00:00
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_read_send ;
req - > async_states - > private_data = io ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_read ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a read and X ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_read_and_X_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_read * io = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
2004-03-23 03:13:24 +00:00
/* readx reply packets can be over-sized */
req - > control_flags | = REQ_CONTROL_LARGE ;
2004-10-26 05:32:11 +00:00
if ( io - > readx . in . maxcnt ! = 0xFFFF & &
io - > readx . in . mincnt ! = 0xFFFF ) {
req_grow_data ( req , 1 + io - > readx . out . nread ) ;
SCVAL ( req - > out . data , 0 , 0 ) ; /* padding */
} else {
req_grow_data ( req , io - > readx . out . nread ) ;
}
2003-08-13 01:53:07 +00:00
/* construct reply */
SSVAL ( req - > out . vwv , VWV ( 0 ) , SMB_CHAIN_NONE ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 2 ) , io - > readx . out . remaining ) ;
SSVAL ( req - > out . vwv , VWV ( 3 ) , io - > readx . out . compaction_mode ) ;
REQ_VWV_RESERVED ( 4 , 1 ) ;
SSVAL ( req - > out . vwv , VWV ( 5 ) , io - > readx . out . nread ) ;
SSVAL ( req - > out . vwv , VWV ( 6 ) , PTR_DIFF ( io - > readx . out . data , req - > out . hdr ) ) ;
REQ_VWV_RESERVED ( 7 , 5 ) ;
2006-03-06 15:00:42 +00:00
smbsrv_chain_reply ( req ) ;
2003-08-13 01:53:07 +00:00
}
/****************************************************************************
Reply to a read and X .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_read_and_X ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_read * io ;
/* parse request */
if ( req - > in . wct ! = 12 ) {
REQ_CHECK_WCT ( req , 10 ) ;
}
REQ_TALLOC ( io , sizeof ( * io ) ) ;
io - > readx . level = RAW_READ_READX ;
io - > readx . in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 2 ) ) ;
io - > readx . in . offset = IVAL ( req - > in . vwv , VWV ( 3 ) ) ;
io - > readx . in . maxcnt = SVAL ( req - > in . vwv , VWV ( 5 ) ) ;
io - > readx . in . mincnt = SVAL ( req - > in . vwv , VWV ( 6 ) ) ;
io - > readx . in . remaining = SVAL ( req - > in . vwv , VWV ( 9 ) ) ;
2004-11-02 22:01:11 +00:00
if ( req - > smb_conn - > negotiate . client_caps & CAP_LARGE_READX ) {
uint32_t high_part = IVAL ( req - > in . vwv , VWV ( 7 ) ) ;
if ( high_part = = 1 ) {
io - > readx . in . maxcnt | = high_part < < 16 ;
}
}
2003-08-13 01:53:07 +00:00
2004-06-27 11:06:10 +00:00
/* the 64 bit variant */
2003-08-13 01:53:07 +00:00
if ( req - > in . wct = = 12 ) {
2004-05-25 16:24:13 +00:00
uint32_t offset_high = IVAL ( req - > in . vwv , VWV ( 10 ) ) ;
2004-05-25 13:57:39 +00:00
io - > readx . in . offset | = ( ( ( uint64_t ) offset_high ) < < 32 ) ;
2003-08-13 01:53:07 +00:00
}
/* setup the reply packet assuming the maximum possible read */
req_setup_reply ( req , 12 , 1 + io - > readx . in . maxcnt ) ;
/* tell the backend where to put the data. Notice the pad byte. */
2004-10-26 05:32:11 +00:00
if ( io - > readx . in . maxcnt ! = 0xFFFF & &
io - > readx . in . mincnt ! = 0xFFFF ) {
io - > readx . out . data = req - > out . data + 1 ;
} else {
io - > readx . out . data = req - > out . data ;
}
2003-08-13 01:53:07 +00:00
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_read_and_X_send ;
req - > async_states - > private_data = io ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_read ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a writebraw ( core + or LANMAN1 .0 protocol ) .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_writebraw ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-12-12 07:22:59 +00:00
req_reply_dos_error ( req , ERRSRV , ERRuseSTD ) ;
2003-08-13 01:53:07 +00:00
}
/****************************************************************************
Reply to a writeunlock ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_writeunlock_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_write * io = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* construct reply */
req_setup_reply ( req , 1 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , io - > writeunlock . out . nwritten ) ;
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a writeunlock ( core + ) .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_writeunlock ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_write * io ;
REQ_CHECK_WCT ( req , 5 ) ;
REQ_TALLOC ( io , sizeof ( * io ) ) ;
io - > writeunlock . level = RAW_WRITE_WRITEUNLOCK ;
io - > writeunlock . in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 0 ) ) ;
io - > writeunlock . in . count = SVAL ( req - > in . vwv , VWV ( 1 ) ) ;
io - > writeunlock . in . offset = IVAL ( req - > in . vwv , VWV ( 2 ) ) ;
io - > writeunlock . in . remaining = SVAL ( req - > in . vwv , VWV ( 4 ) ) ;
io - > writeunlock . in . data = req - > in . data + 3 ;
/* make sure they gave us the data they promised */
if ( io - > writeunlock . in . count + 3 > req - > in . data_size ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
/* make sure the data block is big enough */
if ( SVAL ( req - > in . data , 1 ) < io - > writeunlock . in . count ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_writeunlock_send ;
req - > async_states - > private_data = io ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_write ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a write ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_write_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_write * io = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* construct reply */
req_setup_reply ( req , 1 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , io - > write . out . nwritten ) ;
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a write
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_write ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_write * io ;
REQ_CHECK_WCT ( req , 5 ) ;
REQ_TALLOC ( io , sizeof ( * io ) ) ;
io - > write . level = RAW_WRITE_WRITE ;
io - > write . in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 0 ) ) ;
io - > write . in . count = SVAL ( req - > in . vwv , VWV ( 1 ) ) ;
io - > write . in . offset = IVAL ( req - > in . vwv , VWV ( 2 ) ) ;
io - > write . in . remaining = SVAL ( req - > in . vwv , VWV ( 4 ) ) ;
io - > write . in . data = req - > in . data + 3 ;
/* make sure they gave us the data they promised */
if ( req_data_oob ( req , io - > write . in . data , io - > write . in . count ) ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
/* make sure the data block is big enough */
if ( SVAL ( req - > in . data , 1 ) < io - > write . in . count ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_write_send ;
req - > async_states - > private_data = io ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_write ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a write and X ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_write_and_X_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_write * io = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* construct reply */
req_setup_reply ( req , 6 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , SMB_CHAIN_NONE ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 2 ) , io - > writex . out . nwritten & 0xFFFF ) ;
SSVAL ( req - > out . vwv , VWV ( 3 ) , io - > writex . out . remaining ) ;
SSVAL ( req - > out . vwv , VWV ( 4 ) , io - > writex . out . nwritten > > 16 ) ;
REQ_VWV_RESERVED ( 5 , 1 ) ;
2006-03-06 15:00:42 +00:00
smbsrv_chain_reply ( req ) ;
2003-08-13 01:53:07 +00:00
}
/****************************************************************************
Reply to a write and X .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_write_and_X ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_write * io ;
if ( req - > in . wct ! = 14 ) {
REQ_CHECK_WCT ( req , 12 ) ;
}
REQ_TALLOC ( io , sizeof ( * io ) ) ;
io - > writex . level = RAW_WRITE_WRITEX ;
io - > writex . in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 2 ) ) ;
io - > writex . in . offset = IVAL ( req - > in . vwv , VWV ( 3 ) ) ;
io - > writex . in . wmode = SVAL ( req - > in . vwv , VWV ( 7 ) ) ;
io - > writex . in . remaining = SVAL ( req - > in . vwv , VWV ( 8 ) ) ;
io - > writex . in . count = SVAL ( req - > in . vwv , VWV ( 10 ) ) ;
io - > writex . in . data = req - > in . hdr + SVAL ( req - > in . vwv , VWV ( 11 ) ) ;
if ( req - > in . wct = = 14 ) {
2004-05-25 16:24:13 +00:00
uint32_t offset_high = IVAL ( req - > in . vwv , VWV ( 12 ) ) ;
2004-05-25 17:24:24 +00:00
uint16_t count_high = SVAL ( req - > in . vwv , VWV ( 9 ) ) ;
2004-05-25 13:57:39 +00:00
io - > writex . in . offset | = ( ( ( uint64_t ) offset_high ) < < 32 ) ;
2004-05-25 16:24:13 +00:00
io - > writex . in . count | = ( ( uint32_t ) count_high ) < < 16 ;
2003-08-13 01:53:07 +00:00
}
/* make sure the data is in bounds */
if ( req_data_oob ( req , io - > writex . in . data , io - > writex . in . count ) ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_write_and_X_send ;
req - > async_states - > private_data = io ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_write ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a lseek ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_lseek_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
struct smb_seek * io = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* construct reply */
req_setup_reply ( req , 2 , 0 ) ;
SIVALS ( req - > out . vwv , VWV ( 0 ) , io - > out . offset ) ;
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a lseek .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_lseek ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
struct smb_seek * io ;
REQ_CHECK_WCT ( req , 4 ) ;
REQ_TALLOC ( io , sizeof ( * io ) ) ;
io - > in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 0 ) ) ;
io - > in . mode = SVAL ( req - > in . vwv , VWV ( 1 ) ) ;
io - > in . offset = IVALS ( req - > in . vwv , VWV ( 2 ) ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_lseek_send ;
req - > async_states - > private_data = io ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_seek ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a flush .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_flush ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
struct smb_flush * io ;
/* parse request */
REQ_CHECK_WCT ( req , 1 ) ;
REQ_TALLOC ( io , sizeof ( * io ) ) ;
io - > in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 0 ) ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_simple_send ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_flush ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
2004-09-26 11:30:20 +00:00
Reply to a exit . This closes all files open by a smbpid
2003-08-13 01:53:07 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_exit ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-09-26 11:30:20 +00:00
NTSTATUS status ;
struct smbsrv_tcon * tcon ;
2003-08-13 01:53:07 +00:00
REQ_CHECK_WCT ( req , 0 ) ;
2005-12-06 17:59:20 +00:00
for ( tcon = req - > smb_conn - > smb_tcons . list ; tcon ; tcon = tcon - > next ) {
2004-09-26 11:30:20 +00:00
req - > tcon = tcon ;
2004-09-29 13:17:09 +00:00
status = ntvfs_exit ( req ) ;
2004-09-26 11:30:20 +00:00
req - > tcon = NULL ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
req_reply_error ( req , status ) ;
return ;
}
2003-08-13 02:02:54 +00:00
}
2004-09-26 11:30:20 +00:00
req_setup_reply ( req , 0 , 0 ) ;
req_send_reply ( req ) ;
2003-08-13 01:53:07 +00:00
}
/****************************************************************************
Reply to a close
Note that this has to deal with closing a directory opened by NT SMB ' s .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_close ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_close * io ;
/* parse request */
REQ_CHECK_WCT ( req , 3 ) ;
REQ_TALLOC ( io , sizeof ( * io ) ) ;
io - > close . level = RAW_CLOSE_CLOSE ;
io - > close . in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 0 ) ) ;
2004-06-29 07:40:14 +00:00
io - > close . in . write_time = srv_pull_dos_date3 ( req - > smb_conn , req - > in . vwv + VWV ( 1 ) ) ;
2003-08-13 01:53:07 +00:00
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_simple_send ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_close ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a writeclose ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_writeclose_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_write * io = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* construct reply */
req_setup_reply ( req , 1 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , io - > write . out . nwritten ) ;
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a writeclose ( Core + protocol ) .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_writeclose ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_write * io ;
/* this one is pretty weird - the wct can be 6 or 12 */
if ( req - > in . wct ! = 12 ) {
REQ_CHECK_WCT ( req , 6 ) ;
}
REQ_TALLOC ( io , sizeof ( * io ) ) ;
io - > writeclose . level = RAW_WRITE_WRITECLOSE ;
io - > writeclose . in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 0 ) ) ;
io - > writeclose . in . count = SVAL ( req - > in . vwv , VWV ( 1 ) ) ;
io - > writeclose . in . offset = IVAL ( req - > in . vwv , VWV ( 2 ) ) ;
2004-06-29 07:40:14 +00:00
io - > writeclose . in . mtime = srv_pull_dos_date3 ( req - > smb_conn , req - > in . vwv + VWV ( 4 ) ) ;
2003-08-13 01:53:07 +00:00
io - > writeclose . in . data = req - > in . data + 1 ;
/* make sure they gave us the data they promised */
if ( req_data_oob ( req , io - > writeclose . in . data , io - > writeclose . in . count ) ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_writeclose_send ;
req - > async_states - > private_data = io ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_write ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a lock .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_lock ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_lock * lck ;
/* parse request */
REQ_CHECK_WCT ( req , 5 ) ;
REQ_TALLOC ( lck , sizeof ( * lck ) ) ;
lck - > lock . level = RAW_LOCK_LOCK ;
lck - > lock . in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 0 ) ) ;
lck - > lock . in . count = IVAL ( req - > in . vwv , VWV ( 1 ) ) ;
lck - > lock . in . offset = IVAL ( req - > in . vwv , VWV ( 3 ) ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_simple_send ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_lock ( req , lck ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a unlock .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_unlock ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_lock * lck ;
/* parse request */
REQ_CHECK_WCT ( req , 5 ) ;
REQ_TALLOC ( lck , sizeof ( * lck ) ) ;
lck - > unlock . level = RAW_LOCK_UNLOCK ;
lck - > unlock . in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 0 ) ) ;
lck - > unlock . in . count = IVAL ( req - > in . vwv , VWV ( 1 ) ) ;
lck - > unlock . in . offset = IVAL ( req - > in . vwv , VWV ( 3 ) ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_simple_send ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_lock ( req , lck ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a tdis .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_tdis ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
REQ_CHECK_WCT ( req , 0 ) ;
2004-12-28 23:26:29 +00:00
if ( req - > tcon = = NULL ) {
req_reply_error ( req , NT_STATUS_INVALID_HANDLE ) ;
return ;
}
2005-01-13 18:49:10 +00:00
talloc_free ( req - > tcon ) ;
2003-08-13 01:53:07 +00:00
/* construct reply */
req_setup_reply ( req , 0 , 0 ) ;
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a echo . This is one of the few calls that is handled directly ( the
backends don ' t see it at all )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_echo ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-05-25 17:24:24 +00:00
uint16_t count ;
2003-08-13 01:53:07 +00:00
int i ;
REQ_CHECK_WCT ( req , 0 ) ;
count = SVAL ( req - > in . vwv , VWV ( 0 ) ) ;
req_setup_reply ( req , 1 , req - > in . data_size ) ;
memcpy ( req - > out . data , req - > in . data , req - > in . data_size ) ;
for ( i = 1 ; i < = count ; i + + ) {
2004-10-29 03:48:49 +00:00
struct smbsrv_request * this_req ;
2004-09-25 08:16:16 +00:00
if ( i ! = count ) {
2004-10-29 03:48:49 +00:00
this_req = req_setup_secondary ( req ) ;
} else {
this_req = req ;
2003-08-13 01:53:07 +00:00
}
2004-10-29 03:48:49 +00:00
SSVAL ( this_req - > out . vwv , VWV ( 0 ) , i ) ;
req_send_reply ( this_req ) ;
2003-08-13 01:53:07 +00:00
}
}
/****************************************************************************
Reply to a printopen ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_printopen_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_open * oi = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* construct reply */
req_setup_reply ( req , 1 , 0 ) ;
2004-11-02 04:17:30 +00:00
SSVAL ( req - > out . vwv , VWV ( 0 ) , oi - > openold . out . fnum ) ;
2003-08-13 01:53:07 +00:00
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a printopen .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_printopen ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_open * oi ;
/* parse request */
REQ_CHECK_WCT ( req , 2 ) ;
REQ_TALLOC ( oi , sizeof ( * oi ) ) ;
oi - > splopen . level = RAW_OPEN_SPLOPEN ;
oi - > splopen . in . setup_length = SVAL ( req - > in . vwv , VWV ( 0 ) ) ;
oi - > splopen . in . mode = SVAL ( req - > in . vwv , VWV ( 1 ) ) ;
req_pull_ascii4 ( req , & oi - > splopen . in . ident , req - > in . data , STR_TERMINATE ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_printopen_send ;
req - > async_states - > private_data = oi ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-11-02 04:51:57 +00:00
req - > async_states - > status = ntvfs_openfile ( req , oi ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a printclose .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_printclose ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_close * io ;
/* parse request */
REQ_CHECK_WCT ( req , 3 ) ;
REQ_TALLOC ( io , sizeof ( * io ) ) ;
io - > splclose . level = RAW_CLOSE_SPLCLOSE ;
io - > splclose . in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 0 ) ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_simple_send ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_close ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a printqueue .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
static void reply_printqueue_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_lpq * lpq = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
int i , maxcount ;
const uint_t el_size = 28 ;
CHECK_ASYNC_STATUS ;
/* construct reply */
req_setup_reply ( req , 2 , 0 ) ;
/* truncate the returned list to fit in the negotiated buffer size */
maxcount = ( req_max_data ( req ) - 3 ) / el_size ;
if ( maxcount < lpq - > retq . out . count ) {
lpq - > retq . out . count = maxcount ;
}
/* setup enough space in the reply */
req_grow_data ( req , 3 + el_size * lpq - > retq . out . count ) ;
/* and fill it in */
SSVAL ( req - > out . vwv , VWV ( 0 ) , lpq - > retq . out . count ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , lpq - > retq . out . restart_idx ) ;
SCVAL ( req - > out . data , 0 , SMB_DATA_BLOCK ) ;
SSVAL ( req - > out . data , 1 , el_size * lpq - > retq . out . count ) ;
req - > out . ptr = req - > out . data + 3 ;
for ( i = 0 ; i < lpq - > retq . out . count ; i + + ) {
2004-06-29 07:40:14 +00:00
srv_push_dos_date2 ( req - > smb_conn , req - > out . ptr , 0 , lpq - > retq . out . queue [ i ] . time ) ;
2003-08-13 01:53:07 +00:00
SCVAL ( req - > out . ptr , 4 , lpq - > retq . out . queue [ i ] . status ) ;
SSVAL ( req - > out . ptr , 5 , lpq - > retq . out . queue [ i ] . job ) ;
SIVAL ( req - > out . ptr , 7 , lpq - > retq . out . queue [ i ] . size ) ;
SCVAL ( req - > out . ptr , 11 , 0 ) ; /* reserved */
req_push_str ( req , req - > out . ptr + 12 , lpq - > retq . out . queue [ i ] . user , 16 , STR_ASCII ) ;
req - > out . ptr + = el_size ;
}
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a printqueue .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_printqueue ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_lpq * lpq ;
/* parse request */
REQ_CHECK_WCT ( req , 2 ) ;
REQ_TALLOC ( lpq , sizeof ( * lpq ) ) ;
lpq - > retq . level = RAW_LPQ_RETQ ;
lpq - > retq . in . maxcount = SVAL ( req - > in . vwv , VWV ( 0 ) ) ;
lpq - > retq . in . startidx = SVAL ( req - > in . vwv , VWV ( 1 ) ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_printqueue_send ;
req - > async_states - > private_data = lpq ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_lpq ( req , lpq ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a printwrite .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_printwrite ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_write * io ;
/* parse request */
REQ_CHECK_WCT ( req , 1 ) ;
REQ_TALLOC ( io , sizeof ( * io ) ) ;
io - > splwrite . level = RAW_WRITE_SPLWRITE ;
if ( req - > in . data_size < 3 ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
io - > splwrite . in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 0 ) ) ;
io - > splwrite . in . count = SVAL ( req - > in . data , 1 ) ;
io - > splwrite . in . data = req - > in . data + 3 ;
/* make sure they gave us the data they promised */
if ( req_data_oob ( req , io - > splwrite . in . data , io - > splwrite . in . count ) ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_simple_send ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_write ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a mkdir .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_mkdir ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_mkdir * io ;
/* parse the request */
REQ_CHECK_WCT ( req , 0 ) ;
REQ_TALLOC ( io , sizeof ( * io ) ) ;
io - > generic . level = RAW_MKDIR_MKDIR ;
req_pull_ascii4 ( req , & io - > mkdir . in . path , req - > in . data , STR_TERMINATE ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_simple_send ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_mkdir ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a rmdir .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_rmdir ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
struct smb_rmdir * io ;
/* parse the request */
REQ_CHECK_WCT ( req , 0 ) ;
REQ_TALLOC ( io , sizeof ( * io ) ) ;
req_pull_ascii4 ( req , & io - > in . path , req - > in . data , STR_TERMINATE ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_simple_send ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_rmdir ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a mv .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_mv ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2003-08-13 16:04:21 +00:00
union smb_rename * io ;
2004-12-04 13:56:25 +00:00
uint8_t * p ;
2003-08-13 01:53:07 +00:00
/* parse the request */
REQ_CHECK_WCT ( req , 1 ) ;
REQ_TALLOC ( io , sizeof ( * io ) ) ;
2003-08-13 16:04:21 +00:00
io - > generic . level = RAW_RENAME_RENAME ;
io - > rename . in . attrib = SVAL ( req - > in . vwv , VWV ( 0 ) ) ;
2003-08-13 01:53:07 +00:00
p = req - > in . data ;
2003-08-13 16:04:21 +00:00
p + = req_pull_ascii4 ( req , & io - > rename . in . pattern1 , p , STR_TERMINATE ) ;
p + = req_pull_ascii4 ( req , & io - > rename . in . pattern2 , p , STR_TERMINATE ) ;
2003-08-13 01:53:07 +00:00
2003-08-13 16:04:21 +00:00
if ( ! io - > rename . in . pattern1 | | ! io - > rename . in . pattern2 ) {
2003-08-13 01:53:07 +00:00
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_simple_send ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_rename ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
2003-12-04 02:03:06 +00:00
/****************************************************************************
Reply to an NT rename .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_ntrename ( struct smbsrv_request * req )
2003-12-04 02:03:06 +00:00
{
union smb_rename * io ;
2004-12-04 13:56:25 +00:00
uint8_t * p ;
2003-12-04 02:03:06 +00:00
/* parse the request */
REQ_CHECK_WCT ( req , 4 ) ;
REQ_TALLOC ( io , sizeof ( * io ) ) ;
io - > generic . level = RAW_RENAME_NTRENAME ;
io - > ntrename . in . attrib = SVAL ( req - > in . vwv , VWV ( 0 ) ) ;
io - > ntrename . in . flags = SVAL ( req - > in . vwv , VWV ( 1 ) ) ;
io - > ntrename . in . cluster_size = IVAL ( req - > in . vwv , VWV ( 2 ) ) ;
p = req - > in . data ;
p + = req_pull_ascii4 ( req , & io - > ntrename . in . old_name , p , STR_TERMINATE ) ;
p + = req_pull_ascii4 ( req , & io - > ntrename . in . new_name , p , STR_TERMINATE ) ;
if ( ! io - > ntrename . in . old_name | | ! io - > ntrename . in . new_name ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_simple_send ;
2003-12-04 02:03:06 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_rename ( req , io ) ;
2003-12-04 02:03:06 +00:00
REQ_ASYNC_TAIL ;
}
2003-08-13 01:53:07 +00:00
/****************************************************************************
Reply to a file copy ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_copy_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
struct smb_copy * cp = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* build the reply */
req_setup_reply ( req , 1 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , cp - > out . count ) ;
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a file copy .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_copy ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
struct smb_copy * cp ;
2004-12-04 13:56:25 +00:00
uint8_t * p ;
2003-08-13 01:53:07 +00:00
/* parse request */
REQ_CHECK_WCT ( req , 3 ) ;
REQ_TALLOC ( cp , sizeof ( * cp ) ) ;
cp - > in . tid2 = SVAL ( req - > in . vwv , VWV ( 0 ) ) ;
cp - > in . ofun = SVAL ( req - > in . vwv , VWV ( 1 ) ) ;
cp - > in . flags = SVAL ( req - > in . vwv , VWV ( 2 ) ) ;
p = req - > in . data ;
p + = req_pull_ascii4 ( req , & cp - > in . path1 , p , STR_TERMINATE ) ;
p + = req_pull_ascii4 ( req , & cp - > in . path2 , p , STR_TERMINATE ) ;
if ( ! cp - > in . path1 | | ! cp - > in . path2 ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_copy_send ;
req - > async_states - > private_data = cp ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_copy ( req , cp ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a lockingX request ( async send )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_lockingX_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_lock * lck = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* if it was an oplock break ack then we only send a reply if
there was an error */
if ( lck - > lockx . in . ulock_cnt + lck - > lockx . in . lock_cnt = = 0 ) {
req_destroy ( req ) ;
return ;
}
/* construct reply */
req_setup_reply ( req , 2 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , SMB_CHAIN_NONE ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , 0 ) ;
2006-03-06 15:00:42 +00:00
smbsrv_chain_reply ( req ) ;
2003-08-13 01:53:07 +00:00
}
/****************************************************************************
Reply to a lockingX request .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_lockingX ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_lock * lck ;
uint_t total_locks , i ;
uint_t lck_size ;
2004-12-04 13:56:25 +00:00
uint8_t * p ;
2003-08-13 01:53:07 +00:00
/* parse request */
REQ_CHECK_WCT ( req , 8 ) ;
REQ_TALLOC ( lck , sizeof ( * lck ) ) ;
lck - > lockx . level = RAW_LOCK_LOCKX ;
lck - > lockx . in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 2 ) ) ;
lck - > lockx . in . mode = SVAL ( req - > in . vwv , VWV ( 3 ) ) ;
lck - > lockx . in . timeout = IVAL ( req - > in . vwv , VWV ( 4 ) ) ;
lck - > lockx . in . ulock_cnt = SVAL ( req - > in . vwv , VWV ( 6 ) ) ;
lck - > lockx . in . lock_cnt = SVAL ( req - > in . vwv , VWV ( 7 ) ) ;
total_locks = lck - > lockx . in . ulock_cnt + lck - > lockx . in . lock_cnt ;
2004-06-27 11:06:10 +00:00
/* there are two variants, one with 64 bit offsets and counts */
2003-08-13 01:53:07 +00:00
if ( lck - > lockx . in . mode & LOCKING_ANDX_LARGE_FILES ) {
lck_size = 20 ;
} else {
lck_size = 10 ;
}
/* make sure we got the promised data */
if ( req_data_oob ( req , req - > in . data , total_locks * lck_size ) ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
/* allocate the locks array */
if ( total_locks ) {
REQ_TALLOC ( lck - > lockx . in . locks , total_locks * sizeof ( lck - > lockx . in . locks [ 0 ] ) ) ;
}
p = req - > in . data ;
/* construct the locks array */
for ( i = 0 ; i < total_locks ; i + + ) {
2004-05-25 16:24:13 +00:00
uint32_t ofs_high = 0 , count_high = 0 ;
2003-08-13 01:53:07 +00:00
lck - > lockx . in . locks [ i ] . pid = SVAL ( p , 0 ) ;
if ( lck - > lockx . in . mode & LOCKING_ANDX_LARGE_FILES ) {
ofs_high = IVAL ( p , 4 ) ;
lck - > lockx . in . locks [ i ] . offset = IVAL ( p , 8 ) ;
count_high = IVAL ( p , 12 ) ;
lck - > lockx . in . locks [ i ] . count = IVAL ( p , 16 ) ;
} else {
lck - > lockx . in . locks [ i ] . offset = IVAL ( p , 2 ) ;
lck - > lockx . in . locks [ i ] . count = IVAL ( p , 6 ) ;
}
if ( ofs_high ! = 0 | | count_high ! = 0 ) {
2004-05-25 13:57:39 +00:00
lck - > lockx . in . locks [ i ] . count | = ( ( uint64_t ) count_high ) < < 32 ;
lck - > lockx . in . locks [ i ] . offset | = ( ( uint64_t ) ofs_high ) < < 32 ;
2003-08-13 01:53:07 +00:00
}
p + = lck_size ;
}
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_lockingX_send ;
req - > async_states - > private_data = lck ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_lock ( req , lck ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a SMBreadbmpx ( read block multiplex ) request .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_readbmpx ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
/* tell the client to not use a multiplexed read - its too broken to use */
req_reply_dos_error ( req , ERRSRV , ERRuseSTD ) ;
}
/****************************************************************************
Reply to a SMBsetattrE .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_setattrE ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_setfileinfo * info ;
/* parse request */
REQ_CHECK_WCT ( req , 7 ) ;
REQ_TALLOC ( info , sizeof ( * info ) ) ;
info - > setattre . level = RAW_SFILEINFO_SETATTRE ;
info - > setattre . file . fnum = req_fnum ( req , req - > in . vwv , VWV ( 0 ) ) ;
2004-06-29 07:40:14 +00:00
info - > setattre . in . create_time = srv_pull_dos_date2 ( req - > smb_conn , req - > in . vwv + VWV ( 1 ) ) ;
info - > setattre . in . access_time = srv_pull_dos_date2 ( req - > smb_conn , req - > in . vwv + VWV ( 3 ) ) ;
info - > setattre . in . write_time = srv_pull_dos_date2 ( req - > smb_conn , req - > in . vwv + VWV ( 5 ) ) ;
2003-08-13 01:53:07 +00:00
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_simple_send ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_setfileinfo ( req , info ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to a SMBwritebmpx ( write block multiplex primary ) request .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_writebmpx ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-12-05 11:13:14 +00:00
req_reply_dos_error ( req , ERRSRV , ERRuseSTD ) ;
2003-08-13 01:53:07 +00:00
}
/****************************************************************************
Reply to a SMBwritebs ( write block multiplex secondary ) request .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_writebs ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-12-05 11:13:14 +00:00
req_reply_dos_error ( req , ERRSRV , ERRuseSTD ) ;
2003-08-13 01:53:07 +00:00
}
/****************************************************************************
Reply to a SMBgetattrE ( async reply )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_getattrE_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_fileinfo * info = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* setup reply */
req_setup_reply ( req , 11 , 0 ) ;
2004-06-29 07:40:14 +00:00
srv_push_dos_date2 ( req - > smb_conn , req - > out . vwv , VWV ( 0 ) , info - > getattre . out . create_time ) ;
srv_push_dos_date2 ( req - > smb_conn , req - > out . vwv , VWV ( 2 ) , info - > getattre . out . access_time ) ;
srv_push_dos_date2 ( req - > smb_conn , req - > out . vwv , VWV ( 4 ) , info - > getattre . out . write_time ) ;
2003-08-13 01:53:07 +00:00
SIVAL ( req - > out . vwv , VWV ( 6 ) , info - > getattre . out . size ) ;
SIVAL ( req - > out . vwv , VWV ( 8 ) , info - > getattre . out . alloc_size ) ;
SSVAL ( req - > out . vwv , VWV ( 10 ) , info - > getattre . out . attrib ) ;
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to a SMBgetattrE .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_getattrE ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_fileinfo * info ;
/* parse request */
REQ_CHECK_WCT ( req , 1 ) ;
REQ_TALLOC ( info , sizeof ( * info ) ) ;
info - > getattr . level = RAW_FILEINFO_GETATTRE ;
info - > getattr . in . fnum = req_fnum ( req , req - > in . vwv , VWV ( 0 ) ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_getattrE_send ;
req - > async_states - > private_data = info ;
2003-08-13 01:53:07 +00:00
/* call backend */
2004-10-28 21:48:53 +00:00
req - > async_states - > status = ntvfs_qfileinfo ( req , info ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
reply to an old style session setup command
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_sesssetup_old ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
NTSTATUS status ;
union smb_sesssetup sess ;
2004-12-04 13:56:25 +00:00
uint8_t * p ;
2004-05-25 17:24:24 +00:00
uint16_t passlen ;
2003-08-13 01:53:07 +00:00
sess . old . level = RAW_SESSSETUP_OLD ;
/* parse request */
sess . old . in . bufsize = SVAL ( req - > in . vwv , VWV ( 2 ) ) ;
sess . old . in . mpx_max = SVAL ( req - > in . vwv , VWV ( 3 ) ) ;
sess . old . in . vc_num = SVAL ( req - > in . vwv , VWV ( 4 ) ) ;
sess . old . in . sesskey = IVAL ( req - > in . vwv , VWV ( 5 ) ) ;
passlen = SVAL ( req - > in . vwv , VWV ( 7 ) ) ;
/* check the request isn't malformed */
if ( req_data_oob ( req , req - > in . data , passlen ) ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
p = req - > in . data ;
if ( ! req_pull_blob ( req , p , passlen , & sess . old . in . password ) ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
p + = passlen ;
p + = req_pull_string ( req , & sess . old . in . user , p , - 1 , STR_TERMINATE ) ;
p + = req_pull_string ( req , & sess . old . in . domain , p , - 1 , STR_TERMINATE ) ;
p + = req_pull_string ( req , & sess . old . in . os , p , - 1 , STR_TERMINATE ) ;
p + = req_pull_string ( req , & sess . old . in . lanman , p , - 1 , STR_TERMINATE ) ;
/* call the generic handler */
status = sesssetup_backend ( req , & sess ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
req_reply_error ( req , status ) ;
return ;
}
/* construct reply */
req_setup_reply ( req , 3 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , SMB_CHAIN_NONE ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 2 ) , sess . old . out . action ) ;
SSVAL ( req - > out . hdr , HDR_UID , sess . old . out . vuid ) ;
2006-03-06 15:00:42 +00:00
smbsrv_chain_reply ( req ) ;
2003-08-13 01:53:07 +00:00
}
/****************************************************************************
reply to an NT1 style session setup command
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_sesssetup_nt1 ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
NTSTATUS status ;
union smb_sesssetup sess ;
2004-12-04 13:56:25 +00:00
uint8_t * p ;
2004-05-25 17:24:24 +00:00
uint16_t passlen1 , passlen2 ;
2003-08-13 01:53:07 +00:00
sess . nt1 . level = RAW_SESSSETUP_NT1 ;
/* parse request */
sess . nt1 . in . bufsize = SVAL ( req - > in . vwv , VWV ( 2 ) ) ;
sess . nt1 . in . mpx_max = SVAL ( req - > in . vwv , VWV ( 3 ) ) ;
sess . nt1 . in . vc_num = SVAL ( req - > in . vwv , VWV ( 4 ) ) ;
sess . nt1 . in . sesskey = IVAL ( req - > in . vwv , VWV ( 5 ) ) ;
passlen1 = SVAL ( req - > in . vwv , VWV ( 7 ) ) ;
passlen2 = SVAL ( req - > in . vwv , VWV ( 8 ) ) ;
sess . nt1 . in . capabilities = IVAL ( req - > in . vwv , VWV ( 11 ) ) ;
/* check the request isn't malformed */
if ( req_data_oob ( req , req - > in . data , passlen1 ) | |
req_data_oob ( req , req - > in . data + passlen1 , passlen2 ) ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
p = req - > in . data ;
if ( ! req_pull_blob ( req , p , passlen1 , & sess . nt1 . in . password1 ) ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
p + = passlen1 ;
if ( ! req_pull_blob ( req , p , passlen2 , & sess . nt1 . in . password2 ) ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
p + = passlen2 ;
p + = req_pull_string ( req , & sess . nt1 . in . user , p , - 1 , STR_TERMINATE ) ;
p + = req_pull_string ( req , & sess . nt1 . in . domain , p , - 1 , STR_TERMINATE ) ;
p + = req_pull_string ( req , & sess . nt1 . in . os , p , - 1 , STR_TERMINATE ) ;
p + = req_pull_string ( req , & sess . nt1 . in . lanman , p , - 1 , STR_TERMINATE ) ;
/* call the generic handler */
status = sesssetup_backend ( req , & sess ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
req_reply_error ( req , status ) ;
return ;
}
/* construct reply */
req_setup_reply ( req , 3 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , SMB_CHAIN_NONE ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 2 ) , sess . nt1 . out . action ) ;
SSVAL ( req - > out . hdr , HDR_UID , sess . nt1 . out . vuid ) ;
req_push_str ( req , NULL , sess . nt1 . out . os , - 1 , STR_TERMINATE ) ;
req_push_str ( req , NULL , sess . nt1 . out . lanman , - 1 , STR_TERMINATE ) ;
req_push_str ( req , NULL , sess . nt1 . out . domain , - 1 , STR_TERMINATE ) ;
2006-03-06 15:00:42 +00:00
smbsrv_chain_reply ( req ) ;
2003-08-13 01:53:07 +00:00
}
/****************************************************************************
reply to an SPNEGO style session setup command
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_sesssetup_spnego ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
NTSTATUS status ;
union smb_sesssetup sess ;
2004-12-04 13:56:25 +00:00
uint8_t * p ;
2004-05-25 17:24:24 +00:00
uint16_t blob_len ;
2003-08-13 01:53:07 +00:00
sess . spnego . level = RAW_SESSSETUP_SPNEGO ;
/* parse request */
sess . spnego . in . bufsize = SVAL ( req - > in . vwv , VWV ( 2 ) ) ;
sess . spnego . in . mpx_max = SVAL ( req - > in . vwv , VWV ( 3 ) ) ;
sess . spnego . in . vc_num = SVAL ( req - > in . vwv , VWV ( 4 ) ) ;
sess . spnego . in . sesskey = IVAL ( req - > in . vwv , VWV ( 5 ) ) ;
blob_len = SVAL ( req - > in . vwv , VWV ( 7 ) ) ;
sess . spnego . in . capabilities = IVAL ( req - > in . vwv , VWV ( 10 ) ) ;
p = req - > in . data ;
if ( ! req_pull_blob ( req , p , blob_len , & sess . spnego . in . secblob ) ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
p + = blob_len ;
r6028: A MAJOR update to intergrate the new credentails system fully with
GENSEC, and to pull SCHANNEL into GENSEC, by making it less 'special'.
GENSEC now no longer has it's own handling of 'set username' etc,
instead it uses cli_credentials calls.
In order to link the credentails code right though Samba, a lot of
interfaces have changed to remove 'username, domain, password'
arguments, and these have been replaced with a single 'struct
cli_credentials'.
In the session setup code, a new parameter 'workgroup' contains the
client/server current workgroup, which seems unrelated to the
authentication exchange (it was being filled in from the auth info).
This allows in particular kerberos to only call back for passwords
when it actually needs to perform the kinit.
The kerberos code has been modified not to use the SPNEGO provided
'principal name' (in the mechListMIC), but to instead use the name the
host was connected to as. This better matches Microsoft behaviour,
is more secure and allows better use of standard kerberos functions.
To achieve this, I made changes to our socket code so that the
hostname (before name resolution) is now recorded on the socket.
In schannel, most of the code from librpc/rpc/dcerpc_schannel.c is now
in libcli/auth/schannel.c, and it looks much more like a standard
GENSEC module. The actual sign/seal code moved to
libcli/auth/schannel_sign.c in a previous commit.
The schannel credentails structure is now merged with the rest of the
credentails, as many of the values (username, workstation, domain)
where already present there. This makes handling this in a generic
manner much easier, as there is no longer a custom entry-point.
The auth_domain module continues to be developed, but is now just as
functional as auth_winbind. The changes here are consequential to the
schannel changes.
The only removed function at this point is the RPC-LOGIN test
(simulating the load of a WinXP login), which needs much more work to
clean it up (it contains copies of too much code from all over the
torture suite, and I havn't been able to penetrate its 'structure').
Andrew Bartlett
(This used to be commit 2301a4b38a21aa60917973451687063d83d18d66)
2005-03-24 04:14:06 +00:00
p + = req_pull_string ( req , & sess . spnego . in . os , p , - 1 , STR_TERMINATE ) ;
p + = req_pull_string ( req , & sess . spnego . in . lanman , p , - 1 , STR_TERMINATE ) ;
p + = req_pull_string ( req , & sess . spnego . in . workgroup , p , - 1 , STR_TERMINATE ) ;
2003-08-13 01:53:07 +00:00
/* call the generic handler */
status = sesssetup_backend ( req , & sess ) ;
2004-09-21 01:43:08 +00:00
if ( ! NT_STATUS_IS_OK ( status ) & &
! NT_STATUS_EQUAL ( status , NT_STATUS_MORE_PROCESSING_REQUIRED ) ) {
req_reply_error ( req , status ) ;
return ;
}
2003-08-13 01:53:07 +00:00
/* construct reply */
req_setup_reply ( req , 4 , sess . spnego . out . secblob . length ) ;
2004-09-21 04:11:16 +00:00
if ( NT_STATUS_EQUAL ( status , NT_STATUS_MORE_PROCESSING_REQUIRED ) ) {
req_setup_error ( req , status ) ;
}
2003-08-13 01:53:07 +00:00
SSVAL ( req - > out . vwv , VWV ( 0 ) , SMB_CHAIN_NONE ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 2 ) , sess . spnego . out . action ) ;
SSVAL ( req - > out . vwv , VWV ( 3 ) , sess . spnego . out . secblob . length ) ;
SSVAL ( req - > out . hdr , HDR_UID , sess . spnego . out . vuid ) ;
memcpy ( req - > out . data , sess . spnego . out . secblob . data , sess . spnego . out . secblob . length ) ;
r6028: A MAJOR update to intergrate the new credentails system fully with
GENSEC, and to pull SCHANNEL into GENSEC, by making it less 'special'.
GENSEC now no longer has it's own handling of 'set username' etc,
instead it uses cli_credentials calls.
In order to link the credentails code right though Samba, a lot of
interfaces have changed to remove 'username, domain, password'
arguments, and these have been replaced with a single 'struct
cli_credentials'.
In the session setup code, a new parameter 'workgroup' contains the
client/server current workgroup, which seems unrelated to the
authentication exchange (it was being filled in from the auth info).
This allows in particular kerberos to only call back for passwords
when it actually needs to perform the kinit.
The kerberos code has been modified not to use the SPNEGO provided
'principal name' (in the mechListMIC), but to instead use the name the
host was connected to as. This better matches Microsoft behaviour,
is more secure and allows better use of standard kerberos functions.
To achieve this, I made changes to our socket code so that the
hostname (before name resolution) is now recorded on the socket.
In schannel, most of the code from librpc/rpc/dcerpc_schannel.c is now
in libcli/auth/schannel.c, and it looks much more like a standard
GENSEC module. The actual sign/seal code moved to
libcli/auth/schannel_sign.c in a previous commit.
The schannel credentails structure is now merged with the rest of the
credentails, as many of the values (username, workstation, domain)
where already present there. This makes handling this in a generic
manner much easier, as there is no longer a custom entry-point.
The auth_domain module continues to be developed, but is now just as
functional as auth_winbind. The changes here are consequential to the
schannel changes.
The only removed function at this point is the RPC-LOGIN test
(simulating the load of a WinXP login), which needs much more work to
clean it up (it contains copies of too much code from all over the
torture suite, and I havn't been able to penetrate its 'structure').
Andrew Bartlett
(This used to be commit 2301a4b38a21aa60917973451687063d83d18d66)
2005-03-24 04:14:06 +00:00
req_push_str ( req , NULL , sess . spnego . out . os , - 1 , STR_TERMINATE ) ;
req_push_str ( req , NULL , sess . spnego . out . lanman , - 1 , STR_TERMINATE ) ;
req_push_str ( req , NULL , sess . spnego . out . workgroup , - 1 , STR_TERMINATE ) ;
2003-08-13 01:53:07 +00:00
2006-03-06 15:00:42 +00:00
smbsrv_chain_reply ( req ) ;
2003-08-13 01:53:07 +00:00
}
/****************************************************************************
reply to a session setup command
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_sesssetup ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
switch ( req - > in . wct ) {
case 10 :
/* a pre-NT1 call */
reply_sesssetup_old ( req ) ;
return ;
case 13 :
/* a NT1 call */
reply_sesssetup_nt1 ( req ) ;
return ;
case 12 :
/* a SPNEGO call */
reply_sesssetup_spnego ( req ) ;
return ;
}
2004-06-27 11:06:10 +00:00
/* unsupported variant */
2003-08-13 01:53:07 +00:00
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
}
/****************************************************************************
Reply to a SMBulogoffX .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_ulogoffX ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-09-26 11:30:20 +00:00
struct smbsrv_tcon * tcon ;
NTSTATUS status ;
2003-08-13 01:53:07 +00:00
2005-11-16 18:40:37 +00:00
if ( ! req - > session ) {
2005-04-12 07:29:12 +00:00
req_reply_error ( req , NT_STATUS_DOS ( ERRSRV , ERRbaduid ) ) ;
return ;
}
2004-09-26 11:30:20 +00:00
2003-08-13 01:53:07 +00:00
/* in user level security we are supposed to close any files
2004-09-26 11:30:20 +00:00
open by this user on all open tree connects */
2005-12-06 17:59:20 +00:00
for ( tcon = req - > smb_conn - > smb_tcons . list ; tcon ; tcon = tcon - > next ) {
2005-04-12 07:29:12 +00:00
req - > tcon = tcon ;
status = ntvfs_logoff ( req ) ;
req - > tcon = NULL ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
req_reply_error ( req , status ) ;
return ;
2004-09-26 11:30:20 +00:00
}
2003-08-13 01:53:07 +00:00
}
2005-04-12 07:29:12 +00:00
talloc_free ( req - > session ) ;
req - > session = NULL ; /* it is now invalid, don't use on
any chained packets */
2003-08-13 01:53:07 +00:00
req_setup_reply ( req , 2 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , SMB_CHAIN_NONE ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , 0 ) ;
2006-03-06 15:00:42 +00:00
smbsrv_chain_reply ( req ) ;
2003-08-13 01:53:07 +00:00
}
/****************************************************************************
Reply to an SMBfindclose request
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_findclose ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
NTSTATUS status ;
union smb_search_close io ;
2004-09-21 08:46:47 +00:00
io . findclose . level = RAW_FINDCLOSE_FINDCLOSE ;
2003-08-13 01:53:07 +00:00
/* parse request */
REQ_CHECK_WCT ( req , 1 ) ;
io . findclose . in . handle = SVAL ( req - > in . vwv , VWV ( 0 ) ) ;
/* call backend */
2004-09-29 13:17:09 +00:00
status = ntvfs_search_close ( req , & io ) ;
2003-08-13 01:53:07 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
req_reply_error ( req , status ) ;
return ;
}
/* construct reply */
req_setup_reply ( req , 0 , 0 ) ;
req_send_reply ( req ) ;
}
/****************************************************************************
Reply to an SMBfindnclose request
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_findnclose ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
}
/****************************************************************************
Reply to an SMBntcreateX request ( async send )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-28 08:39:00 +00:00
static void reply_ntcreate_and_X_send ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-10-28 21:48:53 +00:00
union smb_open * io = req - > async_states - > private_data ;
2003-08-13 01:53:07 +00:00
CHECK_ASYNC_STATUS ;
/* construct reply */
req_setup_reply ( req , 34 , 0 ) ;
SSVAL ( req - > out . vwv , VWV ( 0 ) , SMB_CHAIN_NONE ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , 0 ) ;
SCVAL ( req - > out . vwv , VWV ( 2 ) , io - > ntcreatex . out . oplock_level ) ;
/* the rest of the parameters are not aligned! */
SSVAL ( req - > out . vwv , 5 , io - > ntcreatex . out . fnum ) ;
SIVAL ( req - > out . vwv , 7 , io - > ntcreatex . out . create_action ) ;
2004-05-25 13:57:39 +00:00
push_nttime ( req - > out . vwv , 11 , io - > ntcreatex . out . create_time ) ;
push_nttime ( req - > out . vwv , 19 , io - > ntcreatex . out . access_time ) ;
push_nttime ( req - > out . vwv , 27 , io - > ntcreatex . out . write_time ) ;
push_nttime ( req - > out . vwv , 35 , io - > ntcreatex . out . change_time ) ;
2003-08-13 01:53:07 +00:00
SIVAL ( req - > out . vwv , 43 , io - > ntcreatex . out . attrib ) ;
SBVAL ( req - > out . vwv , 47 , io - > ntcreatex . out . alloc_size ) ;
SBVAL ( req - > out . vwv , 55 , io - > ntcreatex . out . size ) ;
SSVAL ( req - > out . vwv , 63 , io - > ntcreatex . out . file_type ) ;
SSVAL ( req - > out . vwv , 65 , io - > ntcreatex . out . ipc_state ) ;
SCVAL ( req - > out . vwv , 67 , io - > ntcreatex . out . is_directory ) ;
2004-12-26 08:13:01 +00:00
req - > chained_fnum = io - > ntcreatex . out . fnum ;
2006-03-06 15:00:42 +00:00
smbsrv_chain_reply ( req ) ;
2003-08-13 01:53:07 +00:00
}
/****************************************************************************
Reply to an SMBntcreateX request
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_ntcreate_and_X ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
union smb_open * io ;
2004-05-25 17:24:24 +00:00
uint16_t fname_len ;
2003-08-13 01:53:07 +00:00
/* parse the request */
REQ_CHECK_WCT ( req , 24 ) ;
REQ_TALLOC ( io , sizeof ( * io ) ) ;
io - > ntcreatex . level = RAW_OPEN_NTCREATEX ;
/* notice that the word parameters are not word aligned, so we don't use VWV() */
fname_len = SVAL ( req - > in . vwv , 5 ) ;
io - > ntcreatex . in . flags = IVAL ( req - > in . vwv , 7 ) ;
io - > ntcreatex . in . root_fid = IVAL ( req - > in . vwv , 11 ) ;
io - > ntcreatex . in . access_mask = IVAL ( req - > in . vwv , 15 ) ;
io - > ntcreatex . in . alloc_size = BVAL ( req - > in . vwv , 19 ) ;
io - > ntcreatex . in . file_attr = IVAL ( req - > in . vwv , 27 ) ;
io - > ntcreatex . in . share_access = IVAL ( req - > in . vwv , 31 ) ;
io - > ntcreatex . in . open_disposition = IVAL ( req - > in . vwv , 35 ) ;
io - > ntcreatex . in . create_options = IVAL ( req - > in . vwv , 39 ) ;
io - > ntcreatex . in . impersonation = IVAL ( req - > in . vwv , 43 ) ;
io - > ntcreatex . in . security_flags = CVAL ( req - > in . vwv , 47 ) ;
2004-11-17 12:36:14 +00:00
io - > ntcreatex . in . ea_list = NULL ;
io - > ntcreatex . in . sec_desc = NULL ;
2003-08-13 01:53:07 +00:00
/* we need a neater way to handle this alignment */
if ( ( req - > flags2 & FLAGS2_UNICODE_STRINGS ) & &
ucs2_align ( req - > in . buffer , req - > in . data , STR_TERMINATE | STR_UNICODE ) ) {
fname_len + + ;
}
req_pull_string ( req , & io - > ntcreatex . in . fname , req - > in . data , fname_len , STR_TERMINATE ) ;
if ( ! io - > ntcreatex . in . fname ) {
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
return ;
}
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_MAY_ASYNC ;
req - > async_states - > send_fn = reply_ntcreate_and_X_send ;
req - > async_states - > private_data = io ;
2003-08-13 01:53:07 +00:00
/* call the backend */
2004-11-02 04:51:57 +00:00
req - > async_states - > status = ntvfs_openfile ( req , io ) ;
2003-08-13 01:53:07 +00:00
REQ_ASYNC_TAIL ;
}
/****************************************************************************
Reply to an SMBntcancel request
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_ntcancel ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-11-04 11:28:38 +00:00
/* NOTE: this request does not generate a reply */
ntvfs_cancel ( req ) ;
req_destroy ( req ) ;
2003-08-13 01:53:07 +00:00
}
/****************************************************************************
Reply to an SMBsends request
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_sends ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
}
/****************************************************************************
Reply to an SMBsendstrt request
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_sendstrt ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
}
/****************************************************************************
Reply to an SMBsendend request
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_sendend ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
}
/****************************************************************************
Reply to an SMBsendtxt request
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_sendtxt ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
req_reply_error ( req , NT_STATUS_FOOBAR ) ;
}
2005-01-22 05:36:32 +00:00
/*
parse the called / calling names from session request
*/
static NTSTATUS parse_session_request ( struct smbsrv_request * req )
{
DATA_BLOB blob ;
NTSTATUS status ;
blob . data = req - > in . buffer + 4 ;
2005-11-10 14:03:07 +00:00
blob . length = ascii_len_n ( ( const char * ) blob . data , req - > in . size - PTR_DIFF ( blob . data , req - > in . buffer ) ) ;
2005-01-22 05:36:32 +00:00
if ( blob . length = = 0 ) return NT_STATUS_BAD_NETWORK_NAME ;
req - > smb_conn - > negotiate . called_name = talloc ( req - > smb_conn , struct nbt_name ) ;
req - > smb_conn - > negotiate . calling_name = talloc ( req - > smb_conn , struct nbt_name ) ;
if ( req - > smb_conn - > negotiate . called_name = = NULL | |
req - > smb_conn - > negotiate . calling_name = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
2005-10-17 15:08:52 +00:00
status = nbt_name_from_blob ( req - > smb_conn , & blob ,
req - > smb_conn - > negotiate . called_name ) ;
2005-01-22 05:36:32 +00:00
NT_STATUS_NOT_OK_RETURN ( status ) ;
blob . data + = blob . length ;
2005-11-10 14:03:07 +00:00
blob . length = ascii_len_n ( ( const char * ) blob . data , req - > in . size - PTR_DIFF ( blob . data , req - > in . buffer ) ) ;
2005-01-22 05:36:32 +00:00
if ( blob . length = = 0 ) return NT_STATUS_BAD_NETWORK_NAME ;
2005-10-17 15:08:52 +00:00
status = nbt_name_from_blob ( req - > smb_conn , & blob ,
req - > smb_conn - > negotiate . calling_name ) ;
2005-01-22 05:36:32 +00:00
NT_STATUS_NOT_OK_RETURN ( status ) ;
req - > smb_conn - > negotiate . done_nbt_session = True ;
return NT_STATUS_OK ;
}
2003-08-13 01:53:07 +00:00
/****************************************************************************
Reply to a special message - a SMB packet with non zero NBT message type
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-03-06 15:31:01 +00:00
void smbsrv_reply_special ( struct smbsrv_request * req )
2003-08-13 01:53:07 +00:00
{
2004-05-25 17:50:17 +00:00
uint8_t msg_type ;
2005-01-07 04:39:16 +00:00
uint8_t * buf = talloc_zero_array ( req , uint8_t , 4 ) ;
2005-01-22 05:36:32 +00:00
2003-08-13 01:53:07 +00:00
msg_type = CVAL ( req - > in . buffer , 0 ) ;
SIVAL ( buf , 0 , 0 ) ;
switch ( msg_type ) {
case 0x81 : /* session request */
2004-06-29 07:40:14 +00:00
if ( req - > smb_conn - > negotiate . done_nbt_session ) {
2004-12-08 11:02:47 +00:00
DEBUG ( 0 , ( " Warning: ignoring secondary session request \n " ) ) ;
2004-12-08 08:09:42 +00:00
return ;
2003-08-13 01:53:07 +00:00
}
SCVAL ( buf , 0 , 0x82 ) ;
SCVAL ( buf , 3 , 0 ) ;
2004-09-22 23:42:20 +00:00
2005-01-22 05:36:32 +00:00
/* we don't check the status - samba always accepts session
requests for any name */
parse_session_request ( req ) ;
2003-08-13 01:53:07 +00:00
req - > out . buffer = buf ;
req - > out . size = 4 ;
2004-05-22 11:16:21 +00:00
req_send_reply_nosign ( req ) ;
2003-08-13 01:53:07 +00:00
return ;
case 0x89 : /* session keepalive request
( some old clients produce this ? ) */
SCVAL ( buf , 0 , SMBkeepalive ) ;
SCVAL ( buf , 3 , 0 ) ;
req - > out . buffer = buf ;
req - > out . size = 4 ;
2004-05-22 11:16:21 +00:00
req_send_reply_nosign ( req ) ;
2003-08-13 01:53:07 +00:00
return ;
case SMBkeepalive :
/* session keepalive - swallow it */
req_destroy ( req ) ;
return ;
}
DEBUG ( 0 , ( " Unexpected NBT session packet (%d) \n " , msg_type ) ) ;
req_destroy ( req ) ;
}