2010-10-19 10:04:27 -07:00
/*
2002-01-30 06:08:46 +00:00
Unix SMB / CIFS implementation .
1998-08-17 06:13:32 +00:00
Manage connections_struct structures
Copyright ( C ) Andrew Tridgell 1998
2002-08-17 15:27:10 +00:00
Copyright ( C ) Alexander Bokovoy 2002
2010-10-19 10:04:27 -07:00
Copyright ( C ) Jeremy Allison 2010
1998-08-17 06:13:32 +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
2007-07-09 19:25:36 +00:00
the Free Software Foundation ; either version 3 of the License , or
1998-08-17 06:13:32 +00:00
( at your option ) any later version .
2010-10-19 10:04:27 -07:00
1998-08-17 06:13:32 +00:00
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 .
2010-10-19 10:04:27 -07:00
1998-08-17 06:13:32 +00:00
You should have received a copy of the GNU General Public License
2007-07-10 00:52:41 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
1998-08-17 06:13:32 +00:00
*/
# include "includes.h"
2011-03-22 16:57:01 +01:00
# include "smbd/smbd.h"
2009-01-08 12:03:45 +01:00
# include "smbd/globals.h"
2011-07-07 21:04:31 +10:00
# include "lib/util/bitmap.h"
1998-08-17 06:13:32 +00:00
2022-07-22 16:28:03 +01:00
static void conn_free_internal ( connection_struct * conn ) ;
/****************************************************************************
* Remove a conn struct from conn - > sconn - > connections
* if not already done .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int conn_struct_destructor ( connection_struct * conn )
{
if ( conn - > sconn ! = NULL ) {
DLIST_REMOVE ( conn - > sconn - > connections , conn ) ;
SMB_ASSERT ( conn - > sconn - > num_connections > 0 ) ;
conn - > sconn - > num_connections - - ;
conn - > sconn = NULL ;
}
conn_free_internal ( conn ) ;
return 0 ;
}
1998-08-17 06:13:32 +00:00
/****************************************************************************
2010-10-19 10:04:27 -07:00
Return the number of open connections .
1998-08-17 06:13:32 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-10-19 10:04:27 -07:00
2009-05-27 11:15:44 +02:00
int conn_num_open ( struct smbd_server_connection * sconn )
1998-08-17 06:13:32 +00:00
{
2012-03-03 05:43:31 +01:00
return sconn - > num_connections ;
1998-08-17 06:13:32 +00:00
}
/****************************************************************************
2010-10-19 10:04:27 -07:00
Check if a snum is in use .
1998-08-17 06:13:32 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-10-19 10:04:27 -07:00
2011-05-25 13:00:22 +10:00
bool conn_snum_used ( struct smbd_server_connection * sconn ,
int snum )
1998-08-17 06:13:32 +00:00
{
2012-03-03 05:43:31 +01:00
struct connection_struct * conn ;
for ( conn = sconn - > connections ; conn ; conn = conn - > next ) {
if ( conn - > params - > service = = snum ) {
return true ;
1998-08-17 06:13:32 +00:00
}
}
2012-03-03 05:43:31 +01:00
2010-10-19 10:12:42 -07:00
return false ;
1998-08-17 06:13:32 +00:00
}
/****************************************************************************
2010-10-19 10:04:27 -07:00
Find first available connection slot , starting from a random position .
2023-07-18 11:30:18 +02:00
The randomisation stops problems with the server dying and clients
2010-10-19 10:04:27 -07:00
thinking the server is still available .
1998-08-17 06:13:32 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-10-19 10:04:27 -07:00
2009-05-27 11:15:44 +02:00
connection_struct * conn_new ( struct smbd_server_connection * sconn )
1998-08-17 06:13:32 +00:00
{
2019-08-15 15:15:30 +02:00
connection_struct * conn = NULL ;
conn = talloc_zero ( NULL , connection_struct ) ;
if ( conn = = NULL ) {
DBG_ERR ( " talloc_zero failed \n " ) ;
return NULL ;
}
conn - > params = talloc ( conn , struct share_params ) ;
if ( conn - > params = = NULL ) {
DBG_ERR ( " talloc_zero failed \n " ) ;
TALLOC_FREE ( conn ) ;
return NULL ;
}
conn - > vuid_cache = talloc_zero ( conn , struct vuid_cache ) ;
if ( conn - > vuid_cache = = NULL ) {
DBG_ERR ( " talloc_zero failed \n " ) ;
TALLOC_FREE ( conn ) ;
return NULL ;
}
conn - > connectpath = talloc_strdup ( conn , " " ) ;
if ( conn - > connectpath = = NULL ) {
DBG_ERR ( " talloc_zero failed \n " ) ;
TALLOC_FREE ( conn ) ;
return NULL ;
}
2019-08-15 15:53:32 +02:00
conn - > cwd_fsp = talloc_zero ( conn , struct files_struct ) ;
if ( conn - > cwd_fsp = = NULL ) {
DBG_ERR ( " talloc_zero failed \n " ) ;
TALLOC_FREE ( conn ) ;
return NULL ;
}
2020-04-14 21:07:24 +02:00
conn - > cwd_fsp - > fsp_name = synthetic_smb_fname ( conn - > cwd_fsp ,
" . " ,
NULL ,
NULL ,
0 ,
0 ) ;
if ( conn - > cwd_fsp - > fsp_name = = NULL ) {
TALLOC_FREE ( conn ) ;
return NULL ;
}
2020-09-27 21:16:03 +02:00
conn - > cwd_fsp - > fh = fd_handle_create ( conn - > cwd_fsp ) ;
2019-08-15 15:53:32 +02:00
if ( conn - > cwd_fsp - > fh = = NULL ) {
DBG_ERR ( " talloc_zero failed \n " ) ;
TALLOC_FREE ( conn ) ;
return NULL ;
}
2009-08-06 13:22:33 +02:00
conn - > sconn = sconn ;
2009-03-03 16:08:56 -08:00
conn - > force_group_gid = ( gid_t ) - 1 ;
2020-09-26 21:46:51 +02:00
fsp_set_fd ( conn - > cwd_fsp , - 1 ) ;
2019-08-15 15:53:32 +02:00
conn - > cwd_fsp - > fnum = FNUM_FIELD_INVALID ;
conn - > cwd_fsp - > conn = conn ;
1998-08-17 06:13:32 +00:00
2012-03-03 05:43:31 +01:00
DLIST_ADD ( sconn - > connections , conn ) ;
sconn - > num_connections + + ;
1998-08-17 06:13:32 +00:00
2022-07-22 16:28:03 +01:00
/*
* Catches the case where someone forgets to call
* conn_free ( ) .
*/
talloc_set_destructor ( conn , conn_struct_destructor ) ;
1998-08-17 06:13:32 +00:00
return conn ;
}
/****************************************************************************
2011-05-25 14:58:24 +10:00
Clear a vuid out of the connection ' s vuid cache
1998-08-17 06:13:32 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-10-19 10:04:27 -07:00
2012-06-05 15:50:21 +02:00
static void conn_clear_vuid_cache ( connection_struct * conn , uint64_t vuid )
1998-08-17 06:13:32 +00:00
{
2011-05-25 14:58:24 +10:00
int i ;
2010-10-19 11:11:56 -07:00
2011-05-25 14:58:24 +10:00
for ( i = 0 ; i < VUID_CACHE_SIZE ; i + + ) {
struct vuid_cache_entry * ent ;
2012-12-20 14:42:55 -08:00
ent = & conn - > vuid_cache - > array [ i ] ;
2011-05-25 14:58:24 +10:00
if ( ent - > vuid = = vuid ) {
ent - > vuid = UID_FIELD_INVALID ;
/*
* We need to keep conn - > session_info around
* if it ' s equal to ent - > session_info as a SMBulogoff
* is often followed by a SMBtdis ( with an invalid
* vuid ) . The debug code ( or regular code in
* vfs_full_audit ) wants to refer to the
* conn - > session_info pointer to print debug
* statements . Theoretically this is a bug ,
* as once the vuid is gone the session_info
* on the conn struct isn ' t valid any more ,
* but there ' s enough code that assumes
* conn - > session_info is never null that
* it ' s easier to hold onto the old pointer
* until we get a new sessionsetupX .
* As everything is hung off the
* conn pointer as a talloc context we ' re not
* leaking memory here . See bug # 6315. JRA .
*/
if ( conn - > session_info = = ent - > session_info ) {
ent - > session_info = NULL ;
} else {
TALLOC_FREE ( ent - > session_info ) ;
2010-10-19 11:11:56 -07:00
}
2011-05-25 14:58:24 +10:00
ent - > read_only = False ;
2012-12-21 09:22:16 -08:00
ent - > share_access = 0 ;
2006-04-15 04:07:10 +00:00
}
1998-08-17 06:13:32 +00:00
}
}
2002-09-25 15:19:00 +00:00
/****************************************************************************
2004-02-13 19:05:25 +00:00
Clear a vuid out of the validity cache , and as the ' owner ' of a connection .
2011-05-25 14:58:24 +10:00
Called from invalidate_vuid ( )
2002-09-25 15:19:00 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-02-13 19:05:25 +00:00
2012-06-05 15:50:21 +02:00
void conn_clear_vuid_caches ( struct smbd_server_connection * sconn , uint64_t vuid )
2002-09-25 15:19:00 +00:00
{
connection_struct * conn ;
2012-03-03 05:43:31 +01:00
for ( conn = sconn - > connections ; conn ; conn = conn - > next ) {
if ( conn - > vuid = = vuid ) {
conn - > vuid = UID_FIELD_INVALID ;
2002-09-25 15:19:00 +00:00
}
2012-03-03 05:43:31 +01:00
conn_clear_vuid_cache ( conn , vuid ) ;
2002-09-25 15:19:00 +00:00
}
}
1998-08-17 06:13:32 +00:00
/****************************************************************************
2005-08-02 23:55:38 +00:00
Free a conn structure - internal part .
1998-08-17 06:13:32 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-08-03 22:38:43 +00:00
2009-08-07 11:48:03 +02:00
static void conn_free_internal ( connection_struct * conn )
1998-08-17 06:13:32 +00:00
{
2009-08-07 11:48:03 +02:00
vfs_handle_struct * handle = NULL , * thandle = NULL ;
2006-12-15 00:49:12 +00:00
struct trans_state * state = NULL ;
2002-08-17 15:27:10 +00:00
2000-04-28 21:09:26 +00:00
/* Free vfs_connection_struct */
2003-05-11 23:34:18 +00:00
handle = conn - > vfs_handles ;
2002-08-17 15:27:10 +00:00
while ( handle ) {
thandle = handle - > next ;
2008-10-01 13:15:54 -07:00
DLIST_REMOVE ( conn - > vfs_handles , handle ) ;
2003-05-11 23:34:18 +00:00
if ( handle - > free_data )
handle - > free_data ( & handle - > data ) ;
2002-08-17 15:27:10 +00:00
handle = thandle ;
2000-04-28 21:09:26 +00:00
}
2006-12-15 00:49:12 +00:00
/* Free any pending transactions stored on this conn. */
for ( state = conn - > pending_trans ; state ; state = state - > next ) {
/* state->setup is a talloc child of state. */
SAFE_FREE ( state - > param ) ;
SAFE_FREE ( state - > data ) ;
}
1998-08-17 06:13:32 +00:00
free_namearray ( conn - > veto_list ) ;
free_namearray ( conn - > hide_list ) ;
free_namearray ( conn - > veto_oplock_list ) ;
2007-10-10 15:34:30 -05:00
free_namearray ( conn - > aio_write_behind_list ) ;
2010-10-19 10:04:27 -07:00
1998-09-05 13:24:20 +00:00
ZERO_STRUCTP ( conn ) ;
1998-08-17 06:13:32 +00:00
}
2001-06-20 03:05:09 +00:00
2005-08-02 23:55:38 +00:00
/****************************************************************************
Free a conn structure .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-08-06 13:22:33 +02:00
void conn_free ( connection_struct * conn )
2005-08-02 23:55:38 +00:00
{
2022-07-22 16:28:03 +01:00
TALLOC_FREE ( conn ) ;
2005-08-02 23:55:38 +00:00
}
2021-01-12 11:39:51 -08:00
/*
* Correctly initialize a share with case options .
*/
void conn_setup_case_options ( connection_struct * conn )
{
int snum = conn - > params - > service ;
if ( lp_case_sensitive ( snum ) = = Auto ) {
/* We will be setting this per packet. Set to be case
* insensitive for now . */
conn - > case_sensitive = false ;
} else {
conn - > case_sensitive = ( bool ) lp_case_sensitive ( snum ) ;
}
conn - > case_preserve = lp_preserve_case ( snum ) ;
conn - > short_case_preserve = lp_short_preserve_case ( snum ) ;
}