2003-08-13 01:53:07 +00:00
/*
Unix SMB / CIFS implementation .
CIFS - on - CIFS NTVFS filesystem backend
Copyright ( C ) Andrew Tridgell 2003
Copyright ( C ) James J Myers 2003 < myersjj @ samba . org >
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
2007-07-10 02:07:03 +00:00
the Free Software Foundation ; either version 3 of the License , or
2003-08-13 01:53:07 +00:00
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
2007-07-10 02:07:03 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2003-08-13 01:53:07 +00:00
*/
/*
this implements a CIFS - > CIFS NTVFS filesystem backend .
*/
# include "includes.h"
2004-11-01 01:03:22 +00:00
# include "libcli/raw/libcliraw.h"
2005-09-26 11:47:55 +00:00
# include "libcli/smb_composite/smb_composite.h"
2005-11-02 00:31:22 +00:00
# include "auth/auth.h"
2006-11-07 00:48:36 +00:00
# include "auth/credentials/credentials.h"
2005-12-27 22:51:30 +00:00
# include "ntvfs/ntvfs.h"
2006-08-30 11:29:34 +00:00
# include "lib/util/dlinklist.h"
2003-08-13 01:53:07 +00:00
2006-05-20 08:15:22 +00:00
struct cvfs_file {
struct cvfs_file * prev , * next ;
uint16_t fnum ;
struct ntvfs_handle * h ;
} ;
2003-08-13 01:53:07 +00:00
/* this is stored in ntvfs_private */
struct cvfs_private {
2004-08-04 13:23:35 +00:00
struct smbcli_tree * tree ;
struct smbcli_transport * transport ;
2006-03-15 17:28:46 +00:00
struct ntvfs_module_context * ntvfs ;
2006-03-08 05:46:09 +00:00
struct async_info * pending ;
2006-05-20 08:15:22 +00:00
struct cvfs_file * files ;
2004-10-20 11:08:58 +00:00
BOOL map_generic ;
2006-05-09 15:50:35 +00:00
BOOL map_trans2 ;
2003-08-13 01:53:07 +00:00
} ;
/* a structure used to pass information to an async handler */
struct async_info {
2006-03-08 05:46:09 +00:00
struct async_info * next , * prev ;
struct cvfs_private * cvfs ;
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req ;
2006-03-08 05:46:09 +00:00
struct smbcli_request * c_req ;
2006-05-20 08:15:22 +00:00
struct cvfs_file * f ;
2003-08-13 01:53:07 +00:00
void * parms ;
} ;
2006-03-16 18:54:19 +00:00
# define SETUP_PID private->tree->session->pid = req->smbpid
2004-10-20 12:08:40 +00:00
2006-05-20 08:15:22 +00:00
# define SETUP_FILE do { \
struct cvfs_file * f ; \
f = ntvfs_handle_get_backend_data ( io - > generic . in . file . ntvfs , ntvfs ) ; \
if ( ! f ) return NT_STATUS_INVALID_HANDLE ; \
io - > generic . in . file . fnum = f - > fnum ; \
} while ( 0 )
# define SETUP_PID_AND_FILE do { \
SETUP_PID ; \
SETUP_FILE ; \
} while ( 0 )
2006-07-23 18:43:07 +00:00
# define CIFS_SERVER "cifs:server"
# define CIFS_USER "cifs:user"
# define CIFS_PASSWORD "cifs:password"
# define CIFS_DOMAIN "cifs:domain"
# define CIFS_SHARE "cifs:share"
# define CIFS_USE_MACHINE_ACCT "cifs:use-machine-account"
# define CIFS_MAP_GENERIC "cifs:map-generic"
# define CIFS_MAP_TRANS2 "cifs:map-trans2"
# define CIFS_USE_MACHINE_ACCT_DEFAULT False
# define CIFS_MAP_GENERIC_DEFAULT False
# define CIFS_MAP_TRANS2_DEFAULT True
2003-08-13 01:53:07 +00:00
/*
a handler for oplock break events from the server - these need to be passed
along to the client
*/
2004-08-04 13:23:35 +00:00
static BOOL oplock_handler ( struct smbcli_transport * transport , uint16_t tid , uint16_t fnum , uint8_t level , void * p_private )
2003-08-13 01:53:07 +00:00
{
struct cvfs_private * private = p_private ;
2006-03-15 17:28:46 +00:00
NTSTATUS status ;
2006-05-20 08:15:22 +00:00
struct ntvfs_handle * h = NULL ;
struct cvfs_file * f ;
for ( f = private - > files ; f ; f = f - > next ) {
if ( f - > fnum ! = fnum ) continue ;
h = f - > h ;
break ;
}
if ( ! h ) {
DEBUG ( 5 , ( " vfs_cifs: ignoring oplock break level %d for fnum %d \n " , level , fnum ) ) ;
return True ;
}
2006-03-15 17:28:46 +00:00
2003-08-13 01:53:07 +00:00
DEBUG ( 5 , ( " vfs_cifs: sending oplock break level %d for fnum %d \n " , level , fnum ) ) ;
2006-05-20 08:15:22 +00:00
status = ntvfs_send_oplock_break ( private - > ntvfs , h , level ) ;
2006-03-15 17:28:46 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) return False ;
return True ;
2003-08-13 01:53:07 +00:00
}
/*
connect to a share - used when a tree_connect operation comes in .
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_connect ( struct ntvfs_module_context * ntvfs ,
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req , const char * sharename )
2003-08-13 01:53:07 +00:00
{
NTSTATUS status ;
struct cvfs_private * private ;
const char * host , * user , * pass , * domain , * remote_share ;
2005-01-23 11:42:56 +00:00
struct smb_composite_connect io ;
2005-01-31 08:30:44 +00:00
struct composite_context * creq ;
2006-07-23 18:43:07 +00:00
struct share_config * scfg = ntvfs - > ctx - > config ;
2003-08-13 01:53:07 +00:00
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
struct cli_credentials * credentials ;
2005-11-05 06:36:42 +00:00
BOOL machine_account ;
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
2003-08-13 01:53:07 +00:00
/* Here we need to determine which server to connect to.
* For now we use parametric options , type cifs .
* Later we will use security = server and auth_server . c .
*/
2006-07-23 18:43:07 +00:00
host = share_string_option ( scfg , CIFS_SERVER , NULL ) ;
user = share_string_option ( scfg , CIFS_USER , NULL ) ;
pass = share_string_option ( scfg , CIFS_PASSWORD , NULL ) ;
domain = share_string_option ( scfg , CIFS_DOMAIN , NULL ) ;
remote_share = share_string_option ( scfg , CIFS_SHARE , NULL ) ;
2003-08-13 01:53:07 +00:00
if ( ! remote_share ) {
remote_share = sharename ;
}
2006-07-23 18:43:07 +00:00
machine_account = share_bool_option ( scfg , CIFS_USE_MACHINE_ACCT , CIFS_USE_MACHINE_ACCT_DEFAULT ) ;
2005-11-05 06:36:42 +00:00
2006-03-08 05:46:09 +00:00
private = talloc_zero ( ntvfs , struct cvfs_private ) ;
2003-08-13 01:53:07 +00:00
if ( ! private ) {
return NT_STATUS_NO_MEMORY ;
}
2004-09-29 13:17:09 +00:00
ntvfs - > private_data = private ;
2003-08-13 01:53:07 +00:00
2005-11-02 00:31:22 +00:00
if ( ! host ) {
DEBUG ( 1 , ( " CIFS backend: You must supply server \n " ) ) ;
return NT_STATUS_INVALID_PARAMETER ;
}
2005-11-05 06:36:42 +00:00
if ( user & & pass ) {
DEBUG ( 5 , ( " CIFS backend: Using specified password \n " ) ) ;
2005-11-02 00:31:22 +00:00
credentials = cli_credentials_init ( private ) ;
2005-11-05 06:36:42 +00:00
if ( ! credentials ) {
return NT_STATUS_NO_MEMORY ;
}
2007-05-25 08:44:33 +00:00
cli_credentials_set_event_context ( credentials , ntvfs - > ctx - > event_ctx ) ;
2005-11-05 06:36:42 +00:00
cli_credentials_set_conf ( credentials ) ;
2005-11-02 00:31:22 +00:00
cli_credentials_set_username ( credentials , user , CRED_SPECIFIED ) ;
2005-11-05 06:36:42 +00:00
if ( domain ) {
cli_credentials_set_domain ( credentials , domain , CRED_SPECIFIED ) ;
}
2005-11-02 00:31:22 +00:00
cli_credentials_set_password ( credentials , pass , CRED_SPECIFIED ) ;
2005-11-05 06:36:42 +00:00
} else if ( machine_account ) {
DEBUG ( 5 , ( " CIFS backend: Using machine account \n " ) ) ;
credentials = cli_credentials_init ( private ) ;
2007-05-25 08:44:33 +00:00
cli_credentials_set_event_context ( credentials , ntvfs - > ctx - > event_ctx ) ;
2005-11-05 06:36:42 +00:00
cli_credentials_set_conf ( credentials ) ;
if ( domain ) {
cli_credentials_set_domain ( credentials , domain , CRED_SPECIFIED ) ;
}
status = cli_credentials_set_machine_account ( credentials ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
2006-03-16 18:54:19 +00:00
} else if ( req - > session_info - > credentials ) {
2005-11-05 06:36:42 +00:00
DEBUG ( 5 , ( " CIFS backend: Using delegated credentials \n " ) ) ;
2006-03-16 18:54:19 +00:00
credentials = req - > session_info - > credentials ;
2005-11-02 00:31:22 +00:00
} else {
2007-05-22 05:22:18 +00:00
DEBUG ( 1 , ( " CIFS backend: NO delegated credentials found: You must supply server, user and password or the client must supply delegated credentials \n " ) ) ;
2005-11-02 00:31:22 +00:00
return NT_STATUS_INVALID_PARAMETER ;
}
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
2005-01-23 11:42:56 +00:00
/* connect to the server, using the smbd event context */
io . in . dest_host = host ;
io . in . port = 0 ;
io . in . called_name = host ;
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
io . in . credentials = credentials ;
2005-10-10 19:57:55 +00:00
io . in . fallback_to_anonymous = False ;
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
io . in . workgroup = lp_workgroup ( ) ;
2005-01-23 11:42:56 +00:00
io . in . service = remote_share ;
io . in . service_type = " ????? " ;
2006-03-15 17:28:46 +00:00
creq = smb_composite_connect_send ( & io , private , ntvfs - > ctx - > event_ctx ) ;
2005-01-23 11:42:56 +00:00
status = smb_composite_connect_recv ( creq , private ) ;
NT_STATUS_NOT_OK_RETURN ( status ) ;
private - > tree = io . out . tree ;
2003-08-13 01:53:07 +00:00
private - > transport = private - > tree - > session - > transport ;
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2006-03-15 17:28:46 +00:00
private - > ntvfs = ntvfs ;
ntvfs - > ctx - > fs_type = talloc_strdup ( ntvfs - > ctx , " NTFS " ) ;
NT_STATUS_HAVE_NO_MEMORY ( ntvfs - > ctx - > fs_type ) ;
ntvfs - > ctx - > dev_type = talloc_strdup ( ntvfs - > ctx , " A: " ) ;
NT_STATUS_HAVE_NO_MEMORY ( ntvfs - > ctx - > dev_type ) ;
2003-08-13 01:53:07 +00:00
/* we need to receive oplock break requests from the server */
2004-08-04 13:23:35 +00:00
smbcli_oplock_handler ( private - > transport , oplock_handler , private ) ;
2004-07-23 06:40:49 +00:00
2006-07-23 18:43:07 +00:00
private - > map_generic = share_bool_option ( scfg , CIFS_MAP_GENERIC , CIFS_MAP_GENERIC_DEFAULT ) ;
2003-08-13 01:53:07 +00:00
2006-07-23 18:43:07 +00:00
private - > map_trans2 = share_bool_option ( scfg , CIFS_MAP_TRANS2 , CIFS_MAP_TRANS2_DEFAULT ) ;
2006-05-09 15:50:35 +00:00
2003-08-13 01:53:07 +00:00
return NT_STATUS_OK ;
}
/*
disconnect from a share
*/
2006-03-10 14:31:17 +00:00
static NTSTATUS cvfs_disconnect ( struct ntvfs_module_context * ntvfs )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2006-03-23 11:13:19 +00:00
struct async_info * a , * an ;
/* first cleanup pending requests */
for ( a = private - > pending ; a ; a = an ) {
an = a - > next ;
smbcli_request_destroy ( a - > c_req ) ;
talloc_free ( a ) ;
}
2003-08-13 01:53:07 +00:00
2004-10-21 06:34:17 +00:00
talloc_free ( private ) ;
2006-01-25 12:19:49 +00:00
ntvfs - > private_data = NULL ;
2003-08-13 01:53:07 +00:00
return NT_STATUS_OK ;
}
2006-03-08 05:46:09 +00:00
/*
destroy an async info structure
*/
2006-05-24 07:34:11 +00:00
static int async_info_destructor ( struct async_info * async )
2006-03-08 05:46:09 +00:00
{
DLIST_REMOVE ( async - > cvfs - > pending , async ) ;
return 0 ;
}
2003-08-13 01:53:07 +00:00
/*
a handler for simple async replies
this handler can only be used for functions that don ' t return any
parameters ( those that just return a status code )
*/
2004-08-04 13:23:35 +00:00
static void async_simple ( struct smbcli_request * c_req )
2003-08-13 01:53:07 +00:00
{
struct async_info * async = c_req - > async . private ;
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req = async - > req ;
2004-10-28 21:48:53 +00:00
req - > async_states - > status = smbcli_request_simple_recv ( c_req ) ;
2006-03-23 11:13:19 +00:00
talloc_free ( async ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > send_fn ( req ) ;
2003-08-13 01:53:07 +00:00
}
/* save some typing for the simple functions */
2006-05-20 08:15:22 +00:00
# define ASYNC_RECV_TAIL_F(io, async_fn, file) do { \
2003-08-13 01:53:07 +00:00
if ( ! c_req ) return NT_STATUS_UNSUCCESSFUL ; \
{ \
2004-01-08 06:48:54 +00:00
struct async_info * async ; \
2005-01-27 07:08:20 +00:00
async = talloc ( req , struct async_info ) ; \
2003-08-13 01:53:07 +00:00
if ( ! async ) return NT_STATUS_NO_MEMORY ; \
async - > parms = io ; \
async - > req = req ; \
2006-05-20 08:15:22 +00:00
async - > f = file ; \
2006-03-08 05:46:09 +00:00
async - > cvfs = private ; \
async - > c_req = c_req ; \
DLIST_ADD ( private - > pending , async ) ; \
2003-08-13 01:53:07 +00:00
c_req - > async . private = async ; \
2006-03-08 05:46:09 +00:00
talloc_set_destructor ( async , async_info_destructor ) ; \
2003-08-13 01:53:07 +00:00
} \
c_req - > async . fn = async_fn ; \
2004-10-28 21:48:53 +00:00
req - > async_states - > state | = NTVFS_ASYNC_STATE_ASYNC ; \
2003-08-13 01:53:07 +00:00
return NT_STATUS_OK ; \
} while ( 0 )
2006-05-20 08:15:22 +00:00
# define ASYNC_RECV_TAIL(io, async_fn) ASYNC_RECV_TAIL_F(io, async_fn, NULL)
2003-08-13 01:53:07 +00:00
# define SIMPLE_ASYNC_TAIL ASYNC_RECV_TAIL(NULL, async_simple)
/*
delete a file - the dirtype specifies the file types to include in the search .
The name can contain CIFS wildcards , but rarely does ( except with OS / 2 clients )
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_unlink ( struct ntvfs_module_context * ntvfs ,
2006-03-10 20:49:20 +00:00
struct ntvfs_request * req , union smb_unlink * unl )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2003-08-13 01:53:07 +00:00
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2003-08-13 01:53:07 +00:00
/* see if the front end will allow us to perform this
function asynchronously . */
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2003-08-13 01:53:07 +00:00
return smb_raw_unlink ( private - > tree , unl ) ;
}
c_req = smb_raw_unlink_send ( private - > tree , unl ) ;
SIMPLE_ASYNC_TAIL ;
}
/*
a handler for async ioctl replies
*/
2004-08-04 13:23:35 +00:00
static void async_ioctl ( struct smbcli_request * c_req )
2003-08-13 01:53:07 +00:00
{
struct async_info * async = c_req - > async . private ;
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req = async - > req ;
2004-10-28 21:48:53 +00:00
req - > async_states - > status = smb_raw_ioctl_recv ( c_req , req , async - > parms ) ;
2006-03-23 11:13:19 +00:00
talloc_free ( async ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > send_fn ( req ) ;
2003-08-13 01:53:07 +00:00
}
/*
ioctl interface
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_ioctl ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req , union smb_ioctl * io )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2003-08-13 01:53:07 +00:00
2006-05-20 08:15:22 +00:00
SETUP_PID_AND_FILE ;
2004-10-20 12:08:40 +00:00
2003-08-13 01:53:07 +00:00
/* see if the front end will allow us to perform this
function asynchronously . */
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2004-09-08 05:39:06 +00:00
return smb_raw_ioctl ( private - > tree , req , io ) ;
2003-08-13 01:53:07 +00:00
}
c_req = smb_raw_ioctl_send ( private - > tree , io ) ;
ASYNC_RECV_TAIL ( io , async_ioctl ) ;
}
/*
check if a directory exists
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_chkpath ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req , union smb_chkpath * cp )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2003-08-13 01:53:07 +00:00
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2003-08-13 01:53:07 +00:00
return smb_raw_chkpath ( private - > tree , cp ) ;
}
c_req = smb_raw_chkpath_send ( private - > tree , cp ) ;
SIMPLE_ASYNC_TAIL ;
}
/*
a handler for async qpathinfo replies
*/
2004-08-04 13:23:35 +00:00
static void async_qpathinfo ( struct smbcli_request * c_req )
2003-08-13 01:53:07 +00:00
{
struct async_info * async = c_req - > async . private ;
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req = async - > req ;
2004-10-28 21:48:53 +00:00
req - > async_states - > status = smb_raw_pathinfo_recv ( c_req , req , async - > parms ) ;
2006-03-23 11:13:19 +00:00
talloc_free ( async ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > send_fn ( req ) ;
2003-08-13 01:53:07 +00:00
}
/*
return info on a pathname
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_qpathinfo ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req , union smb_fileinfo * info )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2003-08-13 01:53:07 +00:00
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2004-09-08 05:39:06 +00:00
return smb_raw_pathinfo ( private - > tree , req , info ) ;
2003-08-13 01:53:07 +00:00
}
c_req = smb_raw_pathinfo_send ( private - > tree , info ) ;
ASYNC_RECV_TAIL ( info , async_qpathinfo ) ;
}
/*
a handler for async qfileinfo replies
*/
2004-08-04 13:23:35 +00:00
static void async_qfileinfo ( struct smbcli_request * c_req )
2003-08-13 01:53:07 +00:00
{
struct async_info * async = c_req - > async . private ;
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req = async - > req ;
2004-10-28 21:48:53 +00:00
req - > async_states - > status = smb_raw_fileinfo_recv ( c_req , req , async - > parms ) ;
2006-03-23 11:13:19 +00:00
talloc_free ( async ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > send_fn ( req ) ;
2003-08-13 01:53:07 +00:00
}
/*
query info on a open file
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_qfileinfo ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req , union smb_fileinfo * io )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2003-08-13 01:53:07 +00:00
2006-05-20 08:15:22 +00:00
SETUP_PID_AND_FILE ;
2004-10-20 12:08:40 +00:00
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2006-05-20 08:15:22 +00:00
return smb_raw_fileinfo ( private - > tree , req , io ) ;
2003-08-13 01:53:07 +00:00
}
2006-05-20 08:15:22 +00:00
c_req = smb_raw_fileinfo_send ( private - > tree , io ) ;
2003-08-13 01:53:07 +00:00
2006-05-20 08:15:22 +00:00
ASYNC_RECV_TAIL ( io , async_qfileinfo ) ;
2003-08-13 01:53:07 +00:00
}
/*
set info on a pathname
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_setpathinfo ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req , union smb_setfileinfo * st )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2003-08-13 01:53:07 +00:00
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2003-08-13 01:53:07 +00:00
return smb_raw_setpathinfo ( private - > tree , st ) ;
}
c_req = smb_raw_setpathinfo_send ( private - > tree , st ) ;
SIMPLE_ASYNC_TAIL ;
}
/*
a handler for async open replies
*/
2004-08-04 13:23:35 +00:00
static void async_open ( struct smbcli_request * c_req )
2003-08-13 01:53:07 +00:00
{
struct async_info * async = c_req - > async . private ;
2006-05-20 08:15:22 +00:00
struct cvfs_private * cvfs = async - > cvfs ;
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req = async - > req ;
2006-05-20 08:15:22 +00:00
struct cvfs_file * f = async - > f ;
union smb_open * io = async - > parms ;
union smb_handle * file ;
2006-03-23 11:13:19 +00:00
talloc_free ( async ) ;
2006-05-20 08:15:22 +00:00
req - > async_states - > status = smb_raw_open_recv ( c_req , req , io ) ;
SMB_OPEN_OUT_FILE ( io , file ) ;
f - > fnum = file - > fnum ;
file - > ntvfs = NULL ;
if ( ! NT_STATUS_IS_OK ( req - > async_states - > status ) ) goto failed ;
req - > async_states - > status = ntvfs_handle_set_backend_data ( f - > h , cvfs - > ntvfs , f ) ;
if ( ! NT_STATUS_IS_OK ( req - > async_states - > status ) ) goto failed ;
file - > ntvfs = f - > h ;
failed :
2004-10-28 21:48:53 +00:00
req - > async_states - > send_fn ( req ) ;
2003-08-13 01:53:07 +00:00
}
/*
open a file
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_open ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req , union smb_open * io )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2006-05-20 08:15:22 +00:00
struct ntvfs_handle * h ;
struct cvfs_file * f ;
NTSTATUS status ;
2003-08-13 01:53:07 +00:00
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2004-10-20 11:08:58 +00:00
if ( io - > generic . level ! = RAW_OPEN_GENERIC & &
private - > map_generic ) {
2006-03-10 14:31:17 +00:00
return ntvfs_map_open ( ntvfs , req , io ) ;
2004-10-20 11:08:58 +00:00
}
2006-05-20 08:15:22 +00:00
status = ntvfs_handle_new ( ntvfs , req , & h ) ;
NT_STATUS_NOT_OK_RETURN ( status ) ;
f = talloc_zero ( h , struct cvfs_file ) ;
NT_STATUS_HAVE_NO_MEMORY ( f ) ;
f - > h = h ;
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2006-05-20 08:15:22 +00:00
union smb_handle * file ;
status = smb_raw_open ( private - > tree , req , io ) ;
NT_STATUS_NOT_OK_RETURN ( status ) ;
SMB_OPEN_OUT_FILE ( io , file ) ;
f - > fnum = file - > fnum ;
file - > ntvfs = NULL ;
status = ntvfs_handle_set_backend_data ( f - > h , private - > ntvfs , f ) ;
NT_STATUS_NOT_OK_RETURN ( status ) ;
file - > ntvfs = f - > h ;
return NT_STATUS_OK ;
2003-08-13 01:53:07 +00:00
}
c_req = smb_raw_open_send ( private - > tree , io ) ;
2006-05-20 08:15:22 +00:00
ASYNC_RECV_TAIL_F ( io , async_open , f ) ;
2003-08-13 01:53:07 +00:00
}
/*
create a directory
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_mkdir ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req , union smb_mkdir * md )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2003-08-13 01:53:07 +00:00
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2003-08-13 01:53:07 +00:00
return smb_raw_mkdir ( private - > tree , md ) ;
}
c_req = smb_raw_mkdir_send ( private - > tree , md ) ;
SIMPLE_ASYNC_TAIL ;
}
/*
remove a directory
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_rmdir ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req , struct smb_rmdir * rd )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2003-08-13 01:53:07 +00:00
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2003-08-13 01:53:07 +00:00
return smb_raw_rmdir ( private - > tree , rd ) ;
}
c_req = smb_raw_rmdir_send ( private - > tree , rd ) ;
SIMPLE_ASYNC_TAIL ;
}
/*
rename a set of files
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_rename ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req , union smb_rename * ren )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2003-08-13 01:53:07 +00:00
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2003-08-13 01:53:07 +00:00
return smb_raw_rename ( private - > tree , ren ) ;
}
c_req = smb_raw_rename_send ( private - > tree , ren ) ;
SIMPLE_ASYNC_TAIL ;
}
/*
copy a set of files
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_copy ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req , struct smb_copy * cp )
2003-08-13 01:53:07 +00:00
{
return NT_STATUS_NOT_SUPPORTED ;
}
/*
a handler for async read replies
*/
2004-08-04 13:23:35 +00:00
static void async_read ( struct smbcli_request * c_req )
2003-08-13 01:53:07 +00:00
{
struct async_info * async = c_req - > async . private ;
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req = async - > req ;
2004-10-28 21:48:53 +00:00
req - > async_states - > status = smb_raw_read_recv ( c_req , async - > parms ) ;
2006-03-23 11:13:19 +00:00
talloc_free ( async ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > send_fn ( req ) ;
2003-08-13 01:53:07 +00:00
}
/*
read from a file
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_read ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req , union smb_read * io )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2003-08-13 01:53:07 +00:00
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2006-05-20 08:15:22 +00:00
if ( io - > generic . level ! = RAW_READ_GENERIC & &
2004-10-20 11:08:58 +00:00
private - > map_generic ) {
2006-05-20 08:15:22 +00:00
return ntvfs_map_read ( ntvfs , req , io ) ;
2004-10-20 11:08:58 +00:00
}
2006-05-20 08:15:22 +00:00
SETUP_FILE ;
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2006-05-20 08:15:22 +00:00
return smb_raw_read ( private - > tree , io ) ;
2003-08-13 01:53:07 +00:00
}
2006-05-20 08:15:22 +00:00
c_req = smb_raw_read_send ( private - > tree , io ) ;
2003-08-13 01:53:07 +00:00
2006-05-20 08:15:22 +00:00
ASYNC_RECV_TAIL ( io , async_read ) ;
2003-08-13 01:53:07 +00:00
}
/*
a handler for async write replies
*/
2004-08-04 13:23:35 +00:00
static void async_write ( struct smbcli_request * c_req )
2003-08-13 01:53:07 +00:00
{
struct async_info * async = c_req - > async . private ;
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req = async - > req ;
2004-10-28 21:48:53 +00:00
req - > async_states - > status = smb_raw_write_recv ( c_req , async - > parms ) ;
2006-03-23 11:13:19 +00:00
talloc_free ( async ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > send_fn ( req ) ;
2003-08-13 01:53:07 +00:00
}
/*
write to a file
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_write ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req , union smb_write * io )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2003-08-13 01:53:07 +00:00
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2006-05-20 08:15:22 +00:00
if ( io - > generic . level ! = RAW_WRITE_GENERIC & &
2004-10-20 11:08:58 +00:00
private - > map_generic ) {
2006-05-20 08:15:22 +00:00
return ntvfs_map_write ( ntvfs , req , io ) ;
2004-10-20 11:08:58 +00:00
}
2006-05-20 08:15:22 +00:00
SETUP_FILE ;
2004-10-20 11:08:58 +00:00
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2006-05-20 08:15:22 +00:00
return smb_raw_write ( private - > tree , io ) ;
2003-08-13 01:53:07 +00:00
}
2006-05-20 08:15:22 +00:00
c_req = smb_raw_write_send ( private - > tree , io ) ;
2003-08-13 01:53:07 +00:00
2006-05-20 08:15:22 +00:00
ASYNC_RECV_TAIL ( io , async_write ) ;
2003-08-13 01:53:07 +00:00
}
2005-06-26 11:34:34 +00:00
/*
a handler for async seek replies
*/
static void async_seek ( struct smbcli_request * c_req )
{
struct async_info * async = c_req - > async . private ;
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req = async - > req ;
2005-06-26 11:34:34 +00:00
req - > async_states - > status = smb_raw_seek_recv ( c_req , async - > parms ) ;
2006-03-23 11:13:19 +00:00
talloc_free ( async ) ;
2005-06-26 11:34:34 +00:00
req - > async_states - > send_fn ( req ) ;
}
2003-08-13 01:53:07 +00:00
/*
seek in a file
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_seek ( struct ntvfs_module_context * ntvfs ,
2006-03-10 20:49:20 +00:00
struct ntvfs_request * req ,
union smb_seek * io )
2003-08-13 01:53:07 +00:00
{
2004-10-20 11:08:58 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
struct smbcli_request * c_req ;
2006-05-20 08:15:22 +00:00
SETUP_PID_AND_FILE ;
2004-10-20 12:08:40 +00:00
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2004-10-20 11:08:58 +00:00
return smb_raw_seek ( private - > tree , io ) ;
}
c_req = smb_raw_seek_send ( private - > tree , io ) ;
2005-06-26 11:34:34 +00:00
ASYNC_RECV_TAIL ( io , async_seek ) ;
2003-08-13 01:53:07 +00:00
}
/*
flush a file
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_flush ( struct ntvfs_module_context * ntvfs ,
2006-03-10 20:49:20 +00:00
struct ntvfs_request * req ,
union smb_flush * io )
2003-08-13 01:53:07 +00:00
{
2004-10-20 11:08:58 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
struct smbcli_request * c_req ;
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2006-05-20 08:15:22 +00:00
switch ( io - > generic . level ) {
case RAW_FLUSH_FLUSH :
SETUP_FILE ;
break ;
case RAW_FLUSH_ALL :
io - > generic . in . file . fnum = 0xFFFF ;
break ;
2006-05-20 10:46:38 +00:00
case RAW_FLUSH_SMB2 :
return NT_STATUS_INVALID_LEVEL ;
2006-05-20 08:15:22 +00:00
}
2004-10-20 12:08:40 +00:00
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2004-10-20 11:08:58 +00:00
return smb_raw_flush ( private - > tree , io ) ;
}
c_req = smb_raw_flush_send ( private - > tree , io ) ;
SIMPLE_ASYNC_TAIL ;
2003-08-13 01:53:07 +00:00
}
/*
close a file
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_close ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req , union smb_close * io )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2003-08-13 01:53:07 +00:00
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2004-10-20 11:08:58 +00:00
if ( io - > generic . level ! = RAW_CLOSE_GENERIC & &
private - > map_generic ) {
2006-03-10 14:31:17 +00:00
return ntvfs_map_close ( ntvfs , req , io ) ;
2004-10-20 11:08:58 +00:00
}
2006-05-20 08:15:22 +00:00
SETUP_FILE ;
2004-10-20 11:08:58 +00:00
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2003-08-13 01:53:07 +00:00
return smb_raw_close ( private - > tree , io ) ;
}
c_req = smb_raw_close_send ( private - > tree , io ) ;
SIMPLE_ASYNC_TAIL ;
}
/*
2004-09-26 11:30:20 +00:00
exit - closing files open by the pid
2003-08-13 01:53:07 +00:00
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_exit ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-09-26 11:30:20 +00:00
struct smbcli_request * c_req ;
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2004-09-26 11:30:20 +00:00
return smb_raw_exit ( private - > tree - > session ) ;
}
c_req = smb_raw_exit_send ( private - > tree - > session ) ;
SIMPLE_ASYNC_TAIL ;
}
/*
logoff - closing files open by the user
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_logoff ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req )
2004-09-26 11:30:20 +00:00
{
/* we can't do this right in the cifs backend .... */
return NT_STATUS_OK ;
2003-08-13 01:53:07 +00:00
}
2004-10-18 13:27:22 +00:00
/*
setup for an async call - nothing to do yet
*/
static NTSTATUS cvfs_async_setup ( struct ntvfs_module_context * ntvfs ,
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req ,
2004-10-18 13:27:22 +00:00
void * private )
{
return NT_STATUS_OK ;
}
2004-11-04 11:28:38 +00:00
/*
cancel an async call
*/
static NTSTATUS cvfs_cancel ( struct ntvfs_module_context * ntvfs ,
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req )
2004-11-04 11:28:38 +00:00
{
2006-03-08 05:46:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2006-03-16 18:54:19 +00:00
struct async_info * a ;
2006-03-08 05:46:09 +00:00
/* find the matching request */
for ( a = private - > pending ; a ; a = a - > next ) {
2006-07-11 18:15:42 +00:00
if ( a - > req = = req ) {
2006-03-08 05:46:09 +00:00
break ;
}
}
if ( a = = NULL ) {
return NT_STATUS_INVALID_PARAMETER ;
}
return smb_raw_ntcancel ( a - > c_req ) ;
2004-11-04 11:28:38 +00:00
}
2003-08-13 01:53:07 +00:00
/*
lock a byte range
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_lock ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req , union smb_lock * io )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2003-08-13 01:53:07 +00:00
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2006-05-20 08:15:22 +00:00
if ( io - > generic . level ! = RAW_LOCK_GENERIC & &
2004-10-20 11:08:58 +00:00
private - > map_generic ) {
2006-05-20 08:15:22 +00:00
return ntvfs_map_lock ( ntvfs , req , io ) ;
2004-10-20 11:08:58 +00:00
}
2006-05-20 08:15:22 +00:00
SETUP_FILE ;
2004-10-20 11:08:58 +00:00
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2006-05-20 08:15:22 +00:00
return smb_raw_lock ( private - > tree , io ) ;
2003-08-13 01:53:07 +00:00
}
2006-05-20 08:15:22 +00:00
c_req = smb_raw_lock_send ( private - > tree , io ) ;
2003-08-13 01:53:07 +00:00
SIMPLE_ASYNC_TAIL ;
}
/*
set info on a open file
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_setfileinfo ( struct ntvfs_module_context * ntvfs ,
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req ,
2006-05-20 08:15:22 +00:00
union smb_setfileinfo * io )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2003-08-13 01:53:07 +00:00
2006-05-20 08:15:22 +00:00
SETUP_PID_AND_FILE ;
2004-10-20 12:08:40 +00:00
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2006-05-20 08:15:22 +00:00
return smb_raw_setfileinfo ( private - > tree , io ) ;
2003-08-13 01:53:07 +00:00
}
2006-05-20 08:15:22 +00:00
c_req = smb_raw_setfileinfo_send ( private - > tree , io ) ;
2003-08-13 01:53:07 +00:00
SIMPLE_ASYNC_TAIL ;
}
/*
a handler for async fsinfo replies
*/
2004-08-04 13:23:35 +00:00
static void async_fsinfo ( struct smbcli_request * c_req )
2003-08-13 01:53:07 +00:00
{
struct async_info * async = c_req - > async . private ;
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req = async - > req ;
2004-10-28 21:48:53 +00:00
req - > async_states - > status = smb_raw_fsinfo_recv ( c_req , req , async - > parms ) ;
2006-03-23 11:13:19 +00:00
talloc_free ( async ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > send_fn ( req ) ;
2003-08-13 01:53:07 +00:00
}
/*
return filesystem space info
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_fsinfo ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req , union smb_fsinfo * fs )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2003-08-13 01:53:07 +00:00
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2004-09-08 05:39:06 +00:00
return smb_raw_fsinfo ( private - > tree , req , fs ) ;
2003-08-13 01:53:07 +00:00
}
2004-09-08 05:39:06 +00:00
c_req = smb_raw_fsinfo_send ( private - > tree , req , fs ) ;
2003-08-13 01:53:07 +00:00
ASYNC_RECV_TAIL ( fs , async_fsinfo ) ;
}
/*
return print queue info
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_lpq ( struct ntvfs_module_context * ntvfs ,
2006-05-20 08:15:22 +00:00
struct ntvfs_request * req , union smb_lpq * lpq )
2003-08-13 01:53:07 +00:00
{
return NT_STATUS_NOT_SUPPORTED ;
}
/*
list files in a directory matching a wildcard pattern
*/
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_search_first ( struct ntvfs_module_context * ntvfs ,
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req , union smb_search_first * io ,
2003-08-13 01:53:07 +00:00
void * search_private ,
2007-05-22 09:25:58 +00:00
BOOL ( * callback ) ( void * , const union smb_search_data * ) )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2003-08-13 01:53:07 +00:00
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2004-09-08 05:39:06 +00:00
return smb_raw_search_first ( private - > tree , req , io , search_private , callback ) ;
2003-08-13 01:53:07 +00:00
}
/* continue a search */
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_search_next ( struct ntvfs_module_context * ntvfs ,
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req , union smb_search_next * io ,
2003-08-13 01:53:07 +00:00
void * search_private ,
2007-05-22 09:25:58 +00:00
BOOL ( * callback ) ( void * , const union smb_search_data * ) )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2003-08-13 01:53:07 +00:00
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2004-09-08 05:39:06 +00:00
return smb_raw_search_next ( private - > tree , req , io , search_private , callback ) ;
2003-08-13 01:53:07 +00:00
}
/* close a search */
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_search_close ( struct ntvfs_module_context * ntvfs ,
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req , union smb_search_close * io )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2003-08-13 01:53:07 +00:00
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2003-08-13 01:53:07 +00:00
return smb_raw_search_close ( private - > tree , io ) ;
}
/*
a handler for async trans2 replies
*/
2004-08-04 13:23:35 +00:00
static void async_trans2 ( struct smbcli_request * c_req )
2003-08-13 01:53:07 +00:00
{
struct async_info * async = c_req - > async . private ;
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req = async - > req ;
2004-10-28 21:48:53 +00:00
req - > async_states - > status = smb_raw_trans2_recv ( c_req , req , async - > parms ) ;
2006-03-23 11:13:19 +00:00
talloc_free ( async ) ;
2004-10-28 21:48:53 +00:00
req - > async_states - > send_fn ( req ) ;
2003-08-13 01:53:07 +00:00
}
/* raw trans2 */
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_trans2 ( struct ntvfs_module_context * ntvfs ,
2006-03-10 20:49:20 +00:00
struct ntvfs_request * req ,
struct smb_trans2 * trans2 )
2003-08-13 01:53:07 +00:00
{
2004-09-29 13:17:09 +00:00
struct cvfs_private * private = ntvfs - > private_data ;
2004-08-04 13:23:35 +00:00
struct smbcli_request * c_req ;
2003-08-13 01:53:07 +00:00
2006-05-18 12:32:23 +00:00
if ( private - > map_trans2 ) {
2006-05-09 15:50:35 +00:00
return NT_STATUS_NOT_IMPLEMENTED ;
}
2004-10-20 12:08:40 +00:00
SETUP_PID ;
2004-10-28 21:48:53 +00:00
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
2004-09-08 05:39:06 +00:00
return smb_raw_trans2 ( private - > tree , req , trans2 ) ;
2003-08-13 01:53:07 +00:00
}
c_req = smb_raw_trans2_send ( private - > tree , trans2 ) ;
ASYNC_RECV_TAIL ( trans2 , async_trans2 ) ;
}
2003-12-11 09:07:45 +00:00
/* SMBtrans - not used on file shares */
2004-09-29 13:17:09 +00:00
static NTSTATUS cvfs_trans ( struct ntvfs_module_context * ntvfs ,
2006-03-10 20:49:20 +00:00
struct ntvfs_request * req ,
struct smb_trans2 * trans2 )
2003-12-11 09:07:45 +00:00
{
return NT_STATUS_ACCESS_DENIED ;
}
2006-03-08 03:54:24 +00:00
/*
a handler for async change notify replies
*/
static void async_changenotify ( struct smbcli_request * c_req )
{
struct async_info * async = c_req - > async . private ;
2006-03-10 14:31:17 +00:00
struct ntvfs_request * req = async - > req ;
2006-03-08 03:54:24 +00:00
req - > async_states - > status = smb_raw_changenotify_recv ( c_req , req , async - > parms ) ;
2006-03-23 11:13:19 +00:00
talloc_free ( async ) ;
2006-03-08 03:54:24 +00:00
req - > async_states - > send_fn ( req ) ;
}
/* change notify request - always async */
static NTSTATUS cvfs_notify ( struct ntvfs_module_context * ntvfs ,
2006-03-10 20:49:20 +00:00
struct ntvfs_request * req ,
2006-07-12 14:25:50 +00:00
union smb_notify * io )
2006-03-08 03:54:24 +00:00
{
struct cvfs_private * private = ntvfs - > private_data ;
struct smbcli_request * c_req ;
2006-03-11 13:06:17 +00:00
int saved_timeout = private - > transport - > options . request_timeout ;
2006-05-20 08:15:22 +00:00
struct cvfs_file * f ;
2006-03-08 03:54:24 +00:00
2006-07-12 14:25:50 +00:00
if ( io - > nttrans . level ! = RAW_NOTIFY_NTTRANS ) {
return NT_STATUS_NOT_IMPLEMENTED ;
}
2006-03-08 03:54:24 +00:00
SETUP_PID ;
2006-07-12 14:25:50 +00:00
f = ntvfs_handle_get_backend_data ( io - > nttrans . in . file . ntvfs , ntvfs ) ;
2006-05-20 08:15:22 +00:00
if ( ! f ) return NT_STATUS_INVALID_HANDLE ;
2006-07-12 14:25:50 +00:00
io - > nttrans . in . file . fnum = f - > fnum ;
2006-05-20 08:15:22 +00:00
2006-03-08 03:54:24 +00:00
/* this request doesn't make sense unless its async */
if ( ! ( req - > async_states - > state & NTVFS_ASYNC_STATE_MAY_ASYNC ) ) {
return NT_STATUS_INVALID_PARAMETER ;
}
2006-03-11 13:06:17 +00:00
/* we must not timeout on notify requests - they wait
forever */
private - > transport - > options . request_timeout = 0 ;
2006-05-20 08:15:22 +00:00
c_req = smb_raw_changenotify_send ( private - > tree , io ) ;
2006-03-08 03:54:24 +00:00
2006-03-11 13:06:17 +00:00
private - > transport - > options . request_timeout = saved_timeout ;
2006-05-20 08:15:22 +00:00
ASYNC_RECV_TAIL ( io , async_changenotify ) ;
2006-03-08 03:54:24 +00:00
}
2003-08-13 01:53:07 +00:00
/*
initialise the CIFS - > CIFS backend , registering ourselves with the ntvfs subsystem
*/
2003-11-25 03:15:26 +00:00
NTSTATUS ntvfs_cifs_init ( void )
2003-08-13 01:53:07 +00:00
{
2003-11-25 03:15:26 +00:00
NTSTATUS ret ;
2003-08-13 01:53:07 +00:00
struct ntvfs_ops ops ;
2006-04-24 01:26:31 +00:00
NTVFS_CURRENT_CRITICAL_SIZES ( vers ) ;
2003-08-13 01:53:07 +00:00
ZERO_STRUCT ( ops ) ;
2003-11-25 03:15:26 +00:00
2004-02-02 13:28:29 +00:00
/* fill in the name and type */
2003-11-25 03:15:26 +00:00
ops . name = " cifs " ;
ops . type = NTVFS_DISK ;
2003-08-13 01:53:07 +00:00
/* fill in all the operations */
ops . connect = cvfs_connect ;
ops . disconnect = cvfs_disconnect ;
ops . unlink = cvfs_unlink ;
ops . chkpath = cvfs_chkpath ;
ops . qpathinfo = cvfs_qpathinfo ;
ops . setpathinfo = cvfs_setpathinfo ;
2006-03-10 14:31:17 +00:00
ops . open = cvfs_open ;
2003-08-13 01:53:07 +00:00
ops . mkdir = cvfs_mkdir ;
ops . rmdir = cvfs_rmdir ;
ops . rename = cvfs_rename ;
ops . copy = cvfs_copy ;
ops . ioctl = cvfs_ioctl ;
ops . read = cvfs_read ;
ops . write = cvfs_write ;
ops . seek = cvfs_seek ;
ops . flush = cvfs_flush ;
ops . close = cvfs_close ;
ops . exit = cvfs_exit ;
ops . lock = cvfs_lock ;
ops . setfileinfo = cvfs_setfileinfo ;
ops . qfileinfo = cvfs_qfileinfo ;
ops . fsinfo = cvfs_fsinfo ;
ops . lpq = cvfs_lpq ;
ops . search_first = cvfs_search_first ;
ops . search_next = cvfs_search_next ;
ops . search_close = cvfs_search_close ;
2003-12-11 09:07:45 +00:00
ops . trans = cvfs_trans ;
2004-09-26 11:30:20 +00:00
ops . logoff = cvfs_logoff ;
2004-10-18 13:27:22 +00:00
ops . async_setup = cvfs_async_setup ;
2004-11-04 11:28:38 +00:00
ops . cancel = cvfs_cancel ;
2006-03-08 03:54:24 +00:00
ops . notify = cvfs_notify ;
2006-05-09 15:50:35 +00:00
ops . trans2 = cvfs_trans2 ;
2003-08-13 01:53:07 +00:00
2004-04-10 20:18:22 +00:00
/* register ourselves with the NTVFS subsystem. We register
under the name ' cifs ' . */
2006-04-24 01:26:31 +00:00
ret = ntvfs_register ( & ops , & vers ) ;
2003-08-13 01:53:07 +00:00
2003-11-25 03:15:26 +00:00
if ( ! NT_STATUS_IS_OK ( ret ) ) {
2003-08-13 01:53:07 +00:00
DEBUG ( 0 , ( " Failed to register CIFS backend! \n " ) ) ;
}
2003-11-25 03:15:26 +00:00
return ret ;
2003-08-13 01:53:07 +00:00
}