2009-01-08 14:03:45 +03:00
/*
Unix SMB / Netbios implementation .
smbd globals
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 18:57:01 +03:00
# include "smbd/smbd.h"
2009-01-08 14:03:45 +03:00
# include "smbd/globals.h"
2010-08-18 14:24:35 +04:00
# include "memcache.h"
2011-03-24 17:31:06 +03:00
# include "messages.h"
2011-05-05 13:25:29 +04:00
# include <tdb.h>
2009-01-08 14:03:45 +03:00
# if defined(WITH_AIO)
struct aio_extra * aio_list_head = NULL ;
2009-01-22 20:04:17 +03:00
struct tevent_signal * aio_signal_event = NULL ;
2009-01-08 14:03:45 +03:00
int aio_pending_size = 0 ;
int outstanding_aio_calls = 0 ;
# endif
# ifdef USE_DMAPI
struct smbd_dmapi_context * dmapi_ctx = NULL ;
# endif
bool dfree_broken = false ;
/* how many write cache buffers have been allocated */
unsigned int allocated_write_caches = 0 ;
const struct mangle_fns * mangle_fns = NULL ;
unsigned char * chartest = NULL ;
TDB_CONTEXT * tdb_mangled_cache = NULL ;
/*
this determines how many characters are used from the original filename
in the 8.3 mangled name . A larger value leads to a weaker hash and more collisions .
The largest possible value is 6.
*/
unsigned mangle_prefix = 0 ;
struct msg_state * smbd_msg_state = NULL ;
bool logged_ioctl_message = false ;
time_t last_smb_conf_reload_time = 0 ;
time_t last_printer_reload_time = 0 ;
/****************************************************************************
structure to hold a linked list of queued messages .
for processing .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
struct pending_message_list * deferred_open_queue = NULL ;
2009-09-17 20:29:07 +04:00
uint32_t common_flags2 = FLAGS2_LONG_PATH_COMPONENTS | FLAGS2_32_BIT_ERROR_CODES | FLAGS2_EXTENDED_ATTRIBUTES ;
2009-01-08 14:03:45 +03:00
struct smb_srv_trans_enc_ctx * partial_srv_trans_enc_ctx = NULL ;
struct smb_srv_trans_enc_ctx * srv_trans_enc_ctx = NULL ;
/* A stack of security contexts. We include the current context as being
the first one , so there is room for another MAX_SEC_CTX_DEPTH more . */
struct sec_ctx sec_ctx_stack [ MAX_SEC_CTX_DEPTH + 1 ] ;
int sec_ctx_stack_ndx = 0 ;
bool become_uid_done = false ;
bool become_gid_done = false ;
connection_struct * last_conn = NULL ;
uint16_t last_flags = 0 ;
uint32_t global_client_caps = 0 ;
uint16_t fnf_handle = 257 ;
/* A stack of current_user connection contexts. */
struct conn_ctx conn_ctx_stack [ MAX_SEC_CTX_DEPTH ] ;
int conn_ctx_stack_ndx = 0 ;
struct vfs_init_function_entry * backends = NULL ;
char * sparse_buf = NULL ;
2010-07-13 20:30:35 +04:00
char * LastDir = NULL ;
2009-01-08 14:03:45 +03:00
/* Current number of oplocks we have outstanding. */
int32_t exclusive_oplocks_open = 0 ;
int32_t level_II_oplocks_open = 0 ;
2009-01-09 16:02:18 +03:00
struct kernel_oplocks * koplocks = NULL ;
2009-01-08 14:03:45 +03:00
int am_parent = 1 ;
struct memcache * smbd_memcache_ctx = NULL ;
bool exit_firsttime = true ;
struct child_pid * children = 0 ;
int num_children = 0 ;
2009-01-08 17:38:47 +03:00
struct smbd_server_connection * smbd_server_conn = NULL ;
2010-09-19 23:09:18 +04:00
struct smbd_server_connection * msg_ctx_to_sconn ( struct messaging_context * msg_ctx )
{
struct server_id my_id , msg_id ;
my_id = messaging_server_id ( smbd_server_conn - > msg_ctx ) ;
msg_id = messaging_server_id ( msg_ctx ) ;
if ( ! procid_equal ( & my_id , & msg_id ) ) {
return NULL ;
}
return smbd_server_conn ;
}
2010-05-26 04:48:15 +04:00
struct messaging_context * smbd_messaging_context ( void )
{
2010-11-14 07:28:41 +03:00
struct messaging_context * msg_ctx = server_messaging_context ( ) ;
if ( likely ( msg_ctx ! = NULL ) ) {
return msg_ctx ;
}
smb_panic ( " Could not init smbd's messaging context. \n " ) ;
2011-03-23 15:00:32 +03:00
return NULL ;
2010-05-26 04:48:15 +04:00
}
struct memcache * smbd_memcache ( void )
{
if ( ! smbd_memcache_ctx ) {
2010-06-11 00:17:35 +04:00
/*
* Note we MUST use the NULL context here , not the
* autofree context , to avoid side effects in forked
* children exiting .
*/
smbd_memcache_ctx = memcache_init ( NULL ,
2010-05-26 04:48:15 +04:00
lp_max_stat_cache_size ( ) * 1024 ) ;
}
if ( ! smbd_memcache_ctx ) {
smb_panic ( " Could not init smbd memcache " ) ;
}
return smbd_memcache_ctx ;
}
2009-01-08 14:03:45 +03:00
void smbd_init_globals ( void )
{
ZERO_STRUCT ( conn_ctx_stack ) ;
ZERO_STRUCT ( sec_ctx_stack ) ;
2009-07-18 04:57:48 +04:00
2011-05-25 12:51:56 +04:00
smbd_server_conn = talloc_zero ( server_event_context ( ) , struct smbd_server_connection ) ;
2009-07-18 04:57:48 +04:00
if ( ! smbd_server_conn ) {
exit_server ( " failed to create smbd_server_connection " ) ;
}
2010-03-22 10:42:13 +03:00
smbd_server_conn - > smb1 . echo_handler . trusted_fd = - 1 ;
smbd_server_conn - > smb1 . echo_handler . socket_lock_fd = - 1 ;
2009-01-08 14:03:45 +03:00
}
2011-05-03 03:32:01 +04:00
void smbd_set_server_fd ( int fd )
{
struct smbd_server_connection * sconn = smbd_server_conn ;
char addr [ INET6_ADDRSTRLEN ] ;
const char * name ;
sconn - > sock = fd ;
/*
* Initialize sconn - > client_id : If we can ' t find the client ' s
* name , default to its address .
*/
client_addr ( fd , sconn - > client_id . addr , sizeof ( sconn - > client_id . addr ) ) ;
name = client_name ( sconn - > sock ) ;
if ( strcmp ( name , " UNKNOWN " ) ! = 0 ) {
name = talloc_strdup ( sconn , name ) ;
} else {
name = NULL ;
}
sconn - > client_id . name =
( name ! = NULL ) ? name : sconn - > client_id . addr ;
sub_set_socket_ids ( sconn - > client_id . addr , sconn - > client_id . name ,
client_socket_addr ( sconn - > sock , addr ,
sizeof ( addr ) ) ) ;
}