2009-05-15 11:50:20 +02:00
/*
Unix SMB / CIFS implementation .
Core SMB2 server
Copyright ( C ) Stefan Metzmacher 2009
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 3 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 , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
2011-03-22 16:57:01 +01:00
# include "smbd/smbd.h"
2009-05-15 11:50:20 +02:00
# include "smbd/globals.h"
2009-08-12 17:52:55 +02:00
# include "../libcli/smb/smb_common.h"
2010-10-12 15:27:50 +11:00
# include "../libcli/security/security.h"
2011-03-24 13:46:20 +01:00
# include "auth.h"
2011-06-29 15:33:54 +10:00
# include "lib/param/loadparm.h"
2012-05-13 17:47:07 +02:00
# include "../lib/util/tevent_ntstatus.h"
2024-09-28 16:48:31 +02:00
# include "smbd/smbXsrv_session.h"
2009-05-15 11:50:20 +02:00
2018-03-21 12:01:05 -07:00
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_SMB2
2012-05-13 17:47:07 +02:00
static struct tevent_req * smbd_smb2_tree_connect_send ( TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
struct smbd_smb2_request * smb2req ,
2018-09-03 15:28:21 +02:00
uint16_t in_flags ,
2012-05-13 17:47:07 +02:00
const char * in_path ) ;
static NTSTATUS smbd_smb2_tree_connect_recv ( struct tevent_req * req ,
uint8_t * out_share_type ,
uint32_t * out_share_flags ,
uint32_t * out_capabilities ,
uint32_t * out_maximal_access ,
2015-05-28 15:35:25 +02:00
uint32_t * out_tree_id ,
bool * disconnect ) ;
2012-05-13 17:47:07 +02:00
2012-05-14 08:41:03 +02:00
static void smbd_smb2_request_tcon_done ( struct tevent_req * subreq ) ;
2009-05-15 11:50:20 +02:00
NTSTATUS smbd_smb2_request_process_tcon ( struct smbd_smb2_request * req )
{
2018-09-03 15:28:21 +02:00
struct smbXsrv_connection * xconn = req - > xconn ;
2009-05-15 11:50:20 +02:00
const uint8_t * inbody ;
2018-09-03 15:28:21 +02:00
uint16_t in_flags ;
2009-05-15 11:50:20 +02:00
uint16_t in_path_offset ;
uint16_t in_path_length ;
DATA_BLOB in_path_buffer ;
char * in_path_string ;
size_t in_path_string_size ;
NTSTATUS status ;
bool ok ;
2012-05-14 08:41:03 +02:00
struct tevent_req * subreq ;
2009-05-15 11:50:20 +02:00
2011-09-06 14:01:43 +02:00
status = smbd_smb2_request_verify_sizes ( req , 0x09 ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return smbd_smb2_request_error ( req , status ) ;
2009-05-15 11:50:20 +02:00
}
2012-08-05 15:00:23 +02:00
inbody = SMBD_SMB2_IN_BODY_PTR ( req ) ;
2009-05-15 11:50:20 +02:00
2018-09-03 15:28:21 +02:00
if ( xconn - > protocol > = PROTOCOL_SMB3_11 ) {
in_flags = SVAL ( inbody , 0x02 ) ;
} else {
in_flags = 0 ;
}
2009-05-15 11:50:20 +02:00
in_path_offset = SVAL ( inbody , 0x04 ) ;
in_path_length = SVAL ( inbody , 0x06 ) ;
2012-08-05 15:00:23 +02:00
if ( in_path_offset ! = ( SMB2_HDR_BODY + SMBD_SMB2_IN_BODY_LEN ( req ) ) ) {
2009-05-15 11:50:20 +02:00
return smbd_smb2_request_error ( req , NT_STATUS_INVALID_PARAMETER ) ;
}
2012-08-05 15:00:23 +02:00
if ( in_path_length > SMBD_SMB2_IN_DYN_LEN ( req ) ) {
2009-05-15 11:50:20 +02:00
return smbd_smb2_request_error ( req , NT_STATUS_INVALID_PARAMETER ) ;
}
2012-08-05 15:00:23 +02:00
in_path_buffer . data = SMBD_SMB2_IN_DYN_PTR ( req ) ;
2009-05-15 11:50:20 +02:00
in_path_buffer . length = in_path_length ;
ok = convert_string_talloc ( req , CH_UTF16 , CH_UNIX ,
in_path_buffer . data ,
in_path_buffer . length ,
& in_path_string ,
2011-03-24 10:59:41 +11:00
& in_path_string_size ) ;
2009-05-15 11:50:20 +02:00
if ( ! ok ) {
return smbd_smb2_request_error ( req , NT_STATUS_ILLEGAL_CHARACTER ) ;
}
2011-09-06 14:14:52 +02:00
if ( in_path_buffer . length = = 0 ) {
in_path_string_size = 0 ;
}
if ( strlen ( in_path_string ) ! = in_path_string_size ) {
return smbd_smb2_request_error ( req , NT_STATUS_BAD_NETWORK_NAME ) ;
}
2012-05-14 08:41:03 +02:00
subreq = smbd_smb2_tree_connect_send ( req ,
2018-12-27 15:18:55 +01:00
req - > sconn - > ev_ctx ,
2012-05-14 08:41:03 +02:00
req ,
2018-09-03 15:28:21 +02:00
in_flags ,
2012-05-14 08:41:03 +02:00
in_path_string ) ;
if ( subreq = = NULL ) {
return smbd_smb2_request_error ( req , NT_STATUS_NO_MEMORY ) ;
}
tevent_req_set_callback ( subreq , smbd_smb2_request_tcon_done , req ) ;
2019-01-24 09:10:11 +01:00
/*
* Avoid sending a STATUS_PENDING message , it ' s very likely
* the client won ' t expect that .
*/
return smbd_smb2_request_pending_queue ( req , subreq , 0 ) ;
2012-05-14 08:41:03 +02:00
}
static void smbd_smb2_request_tcon_done ( struct tevent_req * subreq )
{
struct smbd_smb2_request * req =
tevent_req_callback_data ( subreq ,
struct smbd_smb2_request ) ;
uint8_t * outhdr ;
DATA_BLOB outbody ;
uint8_t out_share_type = 0 ;
uint32_t out_share_flags = 0 ;
uint32_t out_capabilities = 0 ;
uint32_t out_maximal_access = 0 ;
uint32_t out_tree_id = 0 ;
2015-05-28 15:35:25 +02:00
bool disconnect = false ;
2012-05-14 08:41:03 +02:00
NTSTATUS status ;
NTSTATUS error ;
status = smbd_smb2_tree_connect_recv ( subreq ,
& out_share_type ,
& out_share_flags ,
& out_capabilities ,
& out_maximal_access ,
2015-05-28 15:35:25 +02:00
& out_tree_id ,
& disconnect ) ;
2012-05-14 08:41:03 +02:00
TALLOC_FREE ( subreq ) ;
2009-05-15 11:50:20 +02:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2015-05-28 15:35:25 +02:00
if ( disconnect ) {
smbd_server_connection_terminate ( req - > xconn ,
nt_errstr ( status ) ) ;
return ;
}
2012-05-14 08:41:03 +02:00
error = smbd_smb2_request_error ( req , status ) ;
if ( ! NT_STATUS_IS_OK ( error ) ) {
2014-06-11 12:15:48 +02:00
smbd_server_connection_terminate ( req - > xconn ,
2012-05-14 08:41:03 +02:00
nt_errstr ( error ) ) ;
return ;
}
return ;
2009-05-15 11:50:20 +02:00
}
2012-08-05 15:00:23 +02:00
outhdr = SMBD_SMB2_OUT_HDR_PTR ( req ) ;
2009-05-15 11:50:20 +02:00
2013-12-04 14:59:07 +01:00
outbody = smbd_smb2_generate_outbody ( req , 0x10 ) ;
2009-05-15 11:50:20 +02:00
if ( outbody . data = = NULL ) {
2012-05-14 08:41:03 +02:00
error = smbd_smb2_request_error ( req , NT_STATUS_NO_MEMORY ) ;
if ( ! NT_STATUS_IS_OK ( error ) ) {
2014-06-11 12:15:48 +02:00
smbd_server_connection_terminate ( req - > xconn ,
2012-05-14 08:41:03 +02:00
nt_errstr ( error ) ) ;
return ;
}
return ;
2009-05-15 11:50:20 +02:00
}
SIVAL ( outhdr , SMB2_HDR_TID , out_tree_id ) ;
SSVAL ( outbody . data , 0x00 , 0x10 ) ; /* struct size */
2009-06-03 11:31:43 +02:00
SCVAL ( outbody . data , 0x02 ,
out_share_type ) ; /* share type */
2009-05-15 11:50:20 +02:00
SCVAL ( outbody . data , 0x03 , 0 ) ; /* reserved */
2009-06-03 11:31:43 +02:00
SIVAL ( outbody . data , 0x04 ,
out_share_flags ) ; /* share flags */
SIVAL ( outbody . data , 0x08 ,
out_capabilities ) ; /* capabilities */
SIVAL ( outbody . data , 0x0C ,
out_maximal_access ) ; /* maximal access */
2009-05-15 11:50:20 +02:00
2012-05-14 08:41:03 +02:00
error = smbd_smb2_request_done ( req , outbody , NULL ) ;
if ( ! NT_STATUS_IS_OK ( error ) ) {
2014-06-11 12:15:48 +02:00
smbd_server_connection_terminate ( req - > xconn ,
2012-05-14 08:41:03 +02:00
nt_errstr ( error ) ) ;
return ;
}
2009-05-15 11:50:20 +02:00
}
static NTSTATUS smbd_smb2_tree_connect ( struct smbd_smb2_request * req ,
const char * in_path ,
2009-06-03 11:31:43 +02:00
uint8_t * out_share_type ,
uint32_t * out_share_flags ,
uint32_t * out_capabilities ,
uint32_t * out_maximal_access ,
2015-05-28 15:35:25 +02:00
uint32_t * out_tree_id ,
bool * disconnect )
2009-05-15 11:50:20 +02:00
{
2019-10-31 18:56:10 +01:00
const struct loadparm_substitution * lp_sub =
loadparm_s3_global_substitution ( ) ;
2014-06-12 08:38:48 +02:00
struct smbXsrv_connection * conn = req - > xconn ;
2019-12-29 08:31:45 +01:00
struct smbXsrv_session * session = req - > session ;
2020-01-02 17:21:06 +01:00
struct auth_session_info * session_info =
session - > global - > auth_session_info ;
2009-05-15 11:50:20 +02:00
const char * share = in_path ;
2010-11-09 15:07:49 -08:00
char * service = NULL ;
2009-05-15 11:50:20 +02:00
int snum = - 1 ;
2012-03-27 11:09:05 +02:00
struct smbXsrv_tcon * tcon ;
NTTIME now = timeval_to_nttime ( & req - > request_time ) ;
2010-02-24 18:11:07 -08:00
connection_struct * compat_conn = NULL ;
2009-05-27 18:40:42 +02:00
NTSTATUS status ;
2015-11-09 17:23:29 +01:00
bool encryption_desired = req - > session - > global - > encryption_flags & SMBXSRV_ENCRYPTION_DESIRED ;
bool encryption_required = req - > session - > global - > encryption_flags & SMBXSRV_ENCRYPTION_REQUIRED ;
2012-08-08 06:25:10 +02:00
bool guest_session = false ;
2015-05-28 15:35:25 +02:00
bool require_signed_tcon = false ;
2023-04-05 16:59:44 +02:00
uint32_t session_global_id ;
char * share_name = NULL ;
uint8_t encryption_flags = 0 ;
2015-05-28 15:35:25 +02:00
* disconnect = false ;
2009-05-15 11:50:20 +02:00
if ( strncmp ( share , " \\ \\ " , 2 ) = = 0 ) {
const char * p = strchr ( share + 2 , ' \\ ' ) ;
if ( p ) {
share = p + 1 ;
}
}
DEBUG ( 10 , ( " smbd_smb2_tree_connect: path[%s] share[%s] \n " ,
in_path , share ) ) ;
2020-01-02 17:21:06 +01:00
if ( security_session_user_level ( session_info , NULL ) < SECURITY_USER ) {
2015-05-28 15:35:25 +02:00
guest_session = true ;
}
if ( conn - > protocol > = PROTOCOL_SMB3_11 & & ! guest_session ) {
require_signed_tcon = true ;
}
if ( require_signed_tcon & & ! req - > do_encryption & & ! req - > do_signing ) {
DEBUG ( 1 , ( " smbd_smb2_tree_connect: reject request to share "
" [%s] as '%s \\ %s' without encryption or signing. "
" Disconnecting. \n " ,
share ,
req - > session - > global - > auth_session_info - > info - > domain_name ,
req - > session - > global - > auth_session_info - > info - > account_name ) ) ;
* disconnect = true ;
return NT_STATUS_ACCESS_DENIED ;
}
2010-11-09 15:07:49 -08:00
service = talloc_strdup ( talloc_tos ( ) , share ) ;
if ( ! service ) {
return NT_STATUS_NO_MEMORY ;
}
2009-05-15 11:50:20 +02:00
2012-08-08 17:01:00 -07:00
if ( ! strlower_m ( service ) ) {
DEBUG ( 2 , ( " strlower_m %s failed \n " , service ) ) ;
return NT_STATUS_INVALID_PARAMETER ;
}
2009-05-15 11:50:20 +02:00
2010-05-19 21:27:43 -07:00
/* TODO: do more things... */
if ( strequal ( service , HOMES_NAME ) ) {
2019-12-29 08:31:45 +01:00
if ( session - > homes_snum = = - 1 ) {
2010-05-19 21:27:43 -07:00
DEBUG ( 2 , ( " [homes] share not available for "
" user %s because it was not found "
" or created at session setup "
" time \n " ,
2020-01-02 17:21:06 +01:00
session_info - > unix_info - > unix_name ) ) ;
2010-05-19 21:27:43 -07:00
return NT_STATUS_BAD_NETWORK_NAME ;
}
2019-12-29 08:31:45 +01:00
snum = session - > homes_snum ;
} else if ( ( session - > homes_snum ! = - 1 )
2010-05-19 21:27:43 -07:00
& & strequal ( service ,
2019-12-29 08:31:45 +01:00
lp_servicename ( talloc_tos ( ) , lp_sub , session - > homes_snum ) ) ) {
snum = session - > homes_snum ;
2010-05-19 21:27:43 -07:00
} else {
2010-11-09 15:07:49 -08:00
snum = find_service ( talloc_tos ( ) , service , & service ) ;
if ( ! service ) {
return NT_STATUS_NO_MEMORY ;
}
2010-05-19 21:27:43 -07:00
}
2009-05-15 11:50:20 +02:00
if ( snum < 0 ) {
2009-12-02 20:06:37 +01:00
DEBUG ( 3 , ( " smbd_smb2_tree_connect: couldn't find service %s \n " ,
2009-05-15 11:50:20 +02:00
service ) ) ;
return NT_STATUS_BAD_NETWORK_NAME ;
}
2018-03-29 10:00:41 +02:00
/* Handle non-DFS clients attempting connections to msdfs proxy */
if ( lp_host_msdfs ( ) ) {
2019-10-31 18:56:10 +01:00
char * proxy = lp_msdfs_proxy ( talloc_tos ( ) , lp_sub , snum ) ;
2018-03-29 10:00:41 +02:00
if ( ( proxy ! = NULL ) & & ( * proxy ! = ' \0 ' ) ) {
DBG_NOTICE ( " refusing connection to dfs proxy share "
" '%s' (pointing to %s) \n " ,
service ,
proxy ) ;
TALLOC_FREE ( proxy ) ;
return NT_STATUS_BAD_NETWORK_NAME ;
}
TALLOC_FREE ( proxy ) ;
}
2020-05-26 09:34:54 +02:00
if ( ( lp_server_smb_encrypt ( snum ) > = SMB_ENCRYPTION_DESIRED ) & &
2017-01-16 12:56:10 +01:00
( conn - > smb2 . server . cipher ! = 0 ) )
{
2015-07-01 18:07:52 +02:00
encryption_desired = true ;
2015-02-25 16:59:26 +01:00
}
2020-05-26 09:34:54 +02:00
if ( lp_server_smb_encrypt ( snum ) = = SMB_ENCRYPTION_REQUIRED ) {
2015-07-01 18:07:52 +02:00
encryption_desired = true ;
2012-08-08 06:25:10 +02:00
encryption_required = true ;
}
if ( guest_session & & encryption_required ) {
DEBUG ( 1 , ( " reject guest as encryption is required for service %s \n " ,
service ) ) ;
return NT_STATUS_ACCESS_DENIED ;
}
2014-10-13 11:07:01 +02:00
if ( conn - > smb2 . server . cipher = = 0 ) {
2012-08-08 06:25:10 +02:00
if ( encryption_required ) {
DEBUG ( 1 , ( " reject tcon with dialect[0x%04X] "
" as encryption is required for service %s \n " ,
conn - > smb2 . server . dialect , service ) ) ;
return NT_STATUS_ACCESS_DENIED ;
}
2012-07-23 13:47:24 +02:00
}
2023-06-30 18:05:51 +02:00
if ( guest_session ) {
/* make sure we don't ask for optional encryption */
encryption_desired = false ;
}
2015-11-09 17:23:29 +01:00
if ( encryption_desired ) {
2023-04-05 16:59:44 +02:00
encryption_flags | = SMBXSRV_ENCRYPTION_DESIRED ;
2015-11-09 17:23:29 +01:00
}
if ( encryption_required ) {
2023-04-05 16:59:44 +02:00
encryption_flags | = SMBXSRV_ENCRYPTION_REQUIRED ;
}
session_global_id = req - > session - > global - > session_global_id ;
share_name = lp_servicename ( talloc_tos ( ) , lp_sub , snum ) ;
if ( share_name = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
if ( ( lp_max_connections ( snum ) > 0 )
& & ( count_current_connections ( lp_const_servicename ( snum ) , true ) > =
lp_max_connections ( snum ) ) ) {
DBG_WARNING ( " Max connections (%d) exceeded for [%s][%s] \n " ,
lp_max_connections ( snum ) ,
lp_const_servicename ( snum ) , share_name ) ;
TALLOC_FREE ( share_name ) ;
return NT_STATUS_INSUFFICIENT_RESOURCES ;
}
/* create a new tcon as child of the session */
status = smb2srv_tcon_create ( req - > session ,
session_global_id ,
encryption_flags ,
share_name ,
now , & tcon ) ;
TALLOC_FREE ( share_name ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
2015-11-09 17:23:29 +01:00
}
2012-08-08 06:25:10 +02:00
2014-09-15 03:47:41 +02:00
compat_conn = make_connection_smb2 ( req ,
2012-03-27 11:09:05 +02:00
tcon , snum ,
2012-02-03 18:03:10 +11:00
" ??? " ,
2009-05-27 18:40:42 +02:00
& status ) ;
2010-02-24 18:11:07 -08:00
if ( compat_conn = = NULL ) {
2009-05-27 18:40:42 +02:00
TALLOC_FREE ( tcon ) ;
return status ;
}
2012-03-27 11:09:05 +02:00
tcon - > compat = talloc_move ( tcon , & compat_conn ) ;
tcon - > status = NT_STATUS_OK ;
if ( IS_PRINT ( tcon - > compat ) ) {
2010-05-21 16:56:10 -07:00
* out_share_type = SMB2_SHARE_TYPE_PRINT ;
2012-03-27 11:09:05 +02:00
} else if ( IS_IPC ( tcon - > compat ) ) {
2010-05-21 16:56:10 -07:00
* out_share_type = SMB2_SHARE_TYPE_PIPE ;
2010-04-26 21:36:01 -07:00
} else {
2010-05-21 16:56:10 -07:00
* out_share_type = SMB2_SHARE_TYPE_DISK ;
2010-04-26 21:36:01 -07:00
}
2010-05-21 16:56:10 -07:00
2011-11-24 13:49:17 +01:00
* out_share_flags = 0 ;
2010-05-21 16:56:10 -07:00
2012-03-27 11:09:05 +02:00
if ( lp_msdfs_root ( SNUM ( tcon - > compat ) ) & & lp_host_msdfs ( ) ) {
2010-05-21 16:56:10 -07:00
* out_share_flags | = ( SMB2_SHAREFLAG_DFS | SMB2_SHAREFLAG_DFS_ROOT ) ;
* out_capabilities = SMB2_SHARE_CAP_DFS ;
} else {
* out_capabilities = 0 ;
}
2012-03-27 11:09:05 +02:00
switch ( lp_csc_policy ( SNUM ( tcon - > compat ) ) ) {
2010-05-21 16:56:10 -07:00
case CSC_POLICY_MANUAL :
break ;
case CSC_POLICY_DOCUMENTS :
* out_share_flags | = SMB2_SHAREFLAG_AUTO_CACHING ;
break ;
case CSC_POLICY_PROGRAMS :
* out_share_flags | = SMB2_SHAREFLAG_VDO_CACHING ;
break ;
case CSC_POLICY_DISABLE :
* out_share_flags | = SMB2_SHAREFLAG_NO_CACHING ;
break ;
default :
break ;
}
2014-02-04 15:08:58 +13:00
if ( lp_hide_unreadable ( SNUM ( tcon - > compat ) ) | |
2014-02-04 15:08:59 +13:00
lp_hide_unwriteable_files ( SNUM ( tcon - > compat ) ) ) {
2011-11-24 14:42:21 +01:00
* out_share_flags | = SMB2_SHAREFLAG_ACCESS_BASED_DIRECTORY_ENUM ;
}
2015-07-01 18:07:52 +02:00
if ( encryption_desired ) {
2012-08-08 06:25:10 +02:00
* out_share_flags | = SMB2_SHAREFLAG_ENCRYPT_DATA ;
}
2012-07-31 08:55:20 +02:00
/*
* For disk shares we can change the client
* behavior on a cluster . . .
*/
2024-02-08 15:31:10 +01:00
if ( conn - > protocol > = PROTOCOL_SMB3_00 & &
* out_share_type = = SMB2_SHARE_TYPE_DISK )
{
2012-07-31 08:55:20 +02:00
bool persistent = false ; /* persistent handles not implemented yet */
bool cluster = lp_clustering ( ) ;
2024-02-08 15:15:28 +01:00
bool scaleout = cluster ;
bool witness = cluster & & ! lp_rpc_start_on_demand_helpers ( ) ;
2012-07-31 08:55:20 +02:00
bool asymmetric = false ; /* shares are symmetric by default */
bool announce ;
/*
* In a ctdb cluster shares are continuously available ,
* but windows clients mix this with the global persistent
* handles support .
*
* Persistent handles are requested if
* SMB2_SHARE_CAP_CONTINUOUS_AVAILABILITY is present
* even without SMB2_CAP_PERSISTENT_HANDLES .
*
* And SMB2_SHARE_CAP_CONTINUOUS_AVAILABILITY is
* required for SMB2_SHARE_CAP_CLUSTER to have
* an effect .
*
* So we better don ' t announce this by default
* until we support persistent handles .
*/
announce = lp_parm_bool ( SNUM ( tcon - > compat ) ,
" smb3 share cap " ,
" CONTINUOUS AVAILABILITY " ,
persistent ) ;
if ( announce ) {
* out_capabilities | = SMB2_SHARE_CAP_CONTINUOUS_AVAILABILITY ;
}
/*
* ctdb clusters are always scale out . . .
*/
announce = lp_parm_bool ( SNUM ( tcon - > compat ) ,
" smb3 share cap " ,
" SCALE OUT " ,
2024-02-08 15:15:28 +01:00
scaleout ) ;
2012-07-31 08:55:20 +02:00
if ( announce ) {
* out_capabilities | = SMB2_SHARE_CAP_SCALEOUT ;
}
/*
* We support the witness service when ctdb is active
*/
announce = lp_parm_bool ( SNUM ( tcon - > compat ) ,
" smb3 share cap " ,
" CLUSTER " ,
2024-02-08 15:15:28 +01:00
witness ) ;
2012-07-31 08:55:20 +02:00
if ( announce ) {
* out_capabilities | = SMB2_SHARE_CAP_CLUSTER ;
}
/*
* Shares in a ctdb cluster are symmetric by design .
*
* But it might be useful to let the client use
* an isolated transport and witness registration for the
* specific share .
*/
2024-02-08 15:31:10 +01:00
if ( conn - > protocol > = PROTOCOL_SMB3_02 ) {
announce = lp_parm_bool ( SNUM ( tcon - > compat ) ,
" smb3 share cap " ,
" ASYMMETRIC " ,
asymmetric ) ;
}
2012-07-31 08:55:20 +02:00
if ( announce ) {
* out_capabilities | = SMB2_SHARE_CAP_ASYMMETRIC ;
}
}
2012-03-27 11:09:05 +02:00
* out_maximal_access = tcon - > compat - > share_access ;
2009-06-03 11:31:43 +02:00
2012-03-27 11:09:05 +02:00
* out_tree_id = tcon - > global - > tcon_wire_id ;
2017-06-15 23:01:18 +02:00
req - > last_tid = tcon - > global - > tcon_wire_id ;
2009-05-15 11:50:20 +02:00
return NT_STATUS_OK ;
}
2012-05-13 17:47:07 +02:00
struct smbd_smb2_tree_connect_state {
const char * in_path ;
uint8_t out_share_type ;
uint32_t out_share_flags ;
uint32_t out_capabilities ;
uint32_t out_maximal_access ;
uint32_t out_tree_id ;
2015-05-28 15:35:25 +02:00
bool disconnect ;
2012-05-13 17:47:07 +02:00
} ;
static struct tevent_req * smbd_smb2_tree_connect_send ( TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
struct smbd_smb2_request * smb2req ,
2018-09-03 15:28:21 +02:00
uint16_t in_flags ,
2012-05-13 17:47:07 +02:00
const char * in_path )
{
struct tevent_req * req ;
struct smbd_smb2_tree_connect_state * state ;
NTSTATUS status ;
req = tevent_req_create ( mem_ctx , & state ,
struct smbd_smb2_tree_connect_state ) ;
if ( req = = NULL ) {
return NULL ;
}
state - > in_path = in_path ;
status = smbd_smb2_tree_connect ( smb2req ,
state - > in_path ,
& state - > out_share_type ,
& state - > out_share_flags ,
& state - > out_capabilities ,
& state - > out_maximal_access ,
2015-05-28 15:35:25 +02:00
& state - > out_tree_id ,
& state - > disconnect ) ;
2012-05-13 17:47:07 +02:00
if ( tevent_req_nterror ( req , status ) ) {
return tevent_req_post ( req , ev ) ;
}
tevent_req_done ( req ) ;
return tevent_req_post ( req , ev ) ;
}
static NTSTATUS smbd_smb2_tree_connect_recv ( struct tevent_req * req ,
uint8_t * out_share_type ,
uint32_t * out_share_flags ,
uint32_t * out_capabilities ,
uint32_t * out_maximal_access ,
2015-05-28 15:35:25 +02:00
uint32_t * out_tree_id ,
bool * disconnect )
2012-05-13 17:47:07 +02:00
{
struct smbd_smb2_tree_connect_state * state =
tevent_req_data ( req ,
struct smbd_smb2_tree_connect_state ) ;
NTSTATUS status ;
if ( tevent_req_is_nterror ( req , & status ) ) {
tevent_req_received ( req ) ;
return status ;
}
* out_share_type = state - > out_share_type ;
* out_share_flags = state - > out_share_flags ;
* out_capabilities = state - > out_capabilities ;
* out_maximal_access = state - > out_maximal_access ;
* out_tree_id = state - > out_tree_id ;
2015-05-28 15:35:25 +02:00
* disconnect = state - > disconnect ;
2012-05-13 17:47:07 +02:00
tevent_req_received ( req ) ;
return NT_STATUS_OK ;
}
2014-03-10 09:53:18 +01:00
static struct tevent_req * smbd_smb2_tdis_send ( TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
struct smbd_smb2_request * smb2req ) ;
static NTSTATUS smbd_smb2_tdis_recv ( struct tevent_req * req ) ;
static void smbd_smb2_request_tdis_done ( struct tevent_req * subreq ) ;
2009-05-22 12:42:24 +02:00
NTSTATUS smbd_smb2_request_process_tdis ( struct smbd_smb2_request * req )
{
2011-09-06 14:01:43 +02:00
NTSTATUS status ;
2014-03-10 09:53:18 +01:00
struct tevent_req * subreq = NULL ;
2009-05-22 12:42:24 +02:00
2011-09-06 14:01:43 +02:00
status = smbd_smb2_request_verify_sizes ( req , 0x04 ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return smbd_smb2_request_error ( req , status ) ;
2009-05-22 12:42:24 +02:00
}
2018-12-27 15:18:55 +01:00
subreq = smbd_smb2_tdis_send ( req , req - > sconn - > ev_ctx , req ) ;
2014-03-10 09:53:18 +01:00
if ( subreq = = NULL ) {
return smbd_smb2_request_error ( req , NT_STATUS_NO_MEMORY ) ;
}
tevent_req_set_callback ( subreq , smbd_smb2_request_tdis_done , req ) ;
2009-05-22 12:42:24 +02:00
/*
2019-02-12 08:27:43 +01:00
* Avoid sending a STATUS_PENDING message , it ' s very likely
* the client won ' t expect that .
2009-05-22 12:42:24 +02:00
*/
2019-02-12 08:27:43 +01:00
return smbd_smb2_request_pending_queue ( req , subreq , 0 ) ;
2014-03-10 09:53:18 +01:00
}
static void smbd_smb2_request_tdis_done ( struct tevent_req * subreq )
{
struct smbd_smb2_request * smb2req =
tevent_req_callback_data ( subreq ,
struct smbd_smb2_request ) ;
DATA_BLOB outbody ;
NTSTATUS status ;
NTSTATUS error ;
status = smbd_smb2_tdis_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
2012-03-27 11:09:05 +02:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2014-03-10 09:53:18 +01:00
error = smbd_smb2_request_error ( smb2req , status ) ;
if ( ! NT_STATUS_IS_OK ( error ) ) {
2014-06-11 12:15:48 +02:00
smbd_server_connection_terminate ( smb2req - > xconn ,
2014-03-10 09:53:18 +01:00
nt_errstr ( error ) ) ;
return ;
}
return ;
2012-03-27 11:09:05 +02:00
}
2014-03-10 09:53:18 +01:00
outbody = smbd_smb2_generate_outbody ( smb2req , 0x04 ) ;
2009-05-22 12:42:24 +02:00
if ( outbody . data = = NULL ) {
2014-03-10 09:53:18 +01:00
error = smbd_smb2_request_error ( smb2req , NT_STATUS_NO_MEMORY ) ;
if ( ! NT_STATUS_IS_OK ( error ) ) {
2014-06-11 12:15:48 +02:00
smbd_server_connection_terminate ( smb2req - > xconn ,
2014-03-10 09:53:18 +01:00
nt_errstr ( error ) ) ;
return ;
}
return ;
2009-05-22 12:42:24 +02:00
}
SSVAL ( outbody . data , 0x00 , 0x04 ) ; /* struct size */
SSVAL ( outbody . data , 0x02 , 0 ) ; /* reserved */
2014-03-10 09:53:18 +01:00
error = smbd_smb2_request_done ( smb2req , outbody , NULL ) ;
if ( ! NT_STATUS_IS_OK ( error ) ) {
2014-06-11 12:15:48 +02:00
smbd_server_connection_terminate ( smb2req - > xconn ,
2014-03-10 09:53:18 +01:00
nt_errstr ( error ) ) ;
return ;
}
}
struct smbd_smb2_tdis_state {
struct smbd_smb2_request * smb2req ;
2014-03-10 09:53:18 +01:00
struct tevent_queue * wait_queue ;
2014-03-10 09:53:18 +01:00
} ;
2014-03-10 09:53:18 +01:00
static void smbd_smb2_tdis_wait_done ( struct tevent_req * subreq ) ;
2024-09-28 16:48:31 +02:00
struct check_for_lease_break_fsp_cmp_state {
struct smbXsrv_tcon * tcon ;
} ;
static bool check_for_lease_break_fsp_cmp_fn ( struct files_struct * fsp ,
void * private_data )
{
struct check_for_lease_break_fsp_cmp_state * state =
( struct check_for_lease_break_fsp_cmp_state * ) private_data ;
return ( fsp - > conn = = state - > tcon - > compat ) ;
}
2014-03-10 09:53:18 +01:00
static struct tevent_req * smbd_smb2_tdis_send ( TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
struct smbd_smb2_request * smb2req )
{
struct tevent_req * req ;
struct smbd_smb2_tdis_state * state ;
2014-03-10 09:53:18 +01:00
struct tevent_req * subreq ;
2015-05-01 16:50:55 +02:00
struct smbXsrv_connection * xconn = NULL ;
2024-09-28 16:48:31 +02:00
struct check_for_lease_break_fsp_cmp_state fsp_cmp_state ;
2014-03-10 09:53:18 +01:00
req = tevent_req_create ( mem_ctx , & state ,
struct smbd_smb2_tdis_state ) ;
if ( req = = NULL ) {
return NULL ;
}
state - > smb2req = smb2req ;
2014-03-10 09:53:18 +01:00
state - > wait_queue = tevent_queue_create ( state , " tdis_wait_queue " ) ;
if ( tevent_req_nomem ( state - > wait_queue , req ) ) {
return tevent_req_post ( req , ev ) ;
}
/*
* Make sure that no new request will be able to use this tcon .
*/
smb2req - > tcon - > status = NT_STATUS_NETWORK_NAME_DELETED ;
2015-05-01 16:50:55 +02:00
xconn = smb2req - > xconn - > client - > connections ;
for ( ; xconn ! = NULL ; xconn = xconn - > next ) {
struct smbd_smb2_request * preq ;
for ( preq = xconn - > smb2 . requests ; preq ! = NULL ; preq = preq - > next ) {
if ( preq = = smb2req ) {
/* Can't cancel current request. */
continue ;
}
if ( preq - > tcon ! = smb2req - > tcon ) {
/* Request on different tcon. */
continue ;
}
2021-03-23 11:10:22 +01:00
if ( preq - > subreq ! = NULL ) {
2015-05-01 16:50:55 +02:00
tevent_req_cancel ( preq - > subreq ) ;
}
/*
* Now wait until the request is finished .
*
* We don ' t set a callback , as we just want to block the
* wait queue and the talloc_free ( ) of the request will
* remove the item from the wait queue .
*/
subreq = tevent_queue_wait_send ( preq , ev , state - > wait_queue ) ;
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
}
2014-03-10 09:53:18 +01:00
}
}
2024-09-28 16:48:31 +02:00
fsp_cmp_state = ( struct check_for_lease_break_fsp_cmp_state ) {
. tcon = smb2req - > tcon ,
} ;
smbXsrv_wait_for_handle_lease_break ( req ,
ev ,
smb2req - > xconn - > client ,
state - > wait_queue ,
check_for_lease_break_fsp_cmp_fn ,
& fsp_cmp_state ) ;
if ( ! tevent_req_is_in_progress ( req ) ) {
return tevent_req_post ( req , ev ) ;
}
2014-03-10 09:53:18 +01:00
/*
2014-03-10 09:53:18 +01:00
* Now we add our own waiter to the end of the queue ,
* this way we get notified when all pending requests are finished
* and send to the socket .
2014-03-10 09:53:18 +01:00
*/
2014-03-10 09:53:18 +01:00
subreq = tevent_queue_wait_send ( state , ev , state - > wait_queue ) ;
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
}
tevent_req_set_callback ( subreq , smbd_smb2_tdis_wait_done , req ) ;
return req ;
}
static void smbd_smb2_tdis_wait_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct smbd_smb2_tdis_state * state = tevent_req_data (
req , struct smbd_smb2_tdis_state ) ;
NTSTATUS status ;
tevent_queue_wait_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
/*
* As we ' ve been awoken , we may have changed
* uid in the meantime . Ensure we ' re still
* root ( SMB2_OP_TDIS has . as_root = true ) .
*/
change_to_root_user ( ) ;
2014-03-10 09:53:18 +01:00
status = smbXsrv_tcon_disconnect ( state - > smb2req - > tcon ,
state - > smb2req - > tcon - > compat - > vuid ) ;
if ( tevent_req_nterror ( req , status ) ) {
2014-03-10 09:53:18 +01:00
return ;
2014-03-10 09:53:18 +01:00
}
/* We did tear down the tcon. */
TALLOC_FREE ( state - > smb2req - > tcon ) ;
tevent_req_done ( req ) ;
}
static NTSTATUS smbd_smb2_tdis_recv ( struct tevent_req * req )
{
return tevent_req_simple_recv_ntstatus ( req ) ;
2009-05-22 12:42:24 +02:00
}