mirror of
https://github.com/samba-team/samba.git
synced 2025-01-14 19:24:43 +03:00
84dad60672
As part of forcibly disconnecting a client from a share, smbd must atomically call reload_services() to ensure that the entry in the ServicePtrs[] array corresponding to that share is removed if the share was removed from the smb.conf or registry entries. Otherwise the ServicePtrs[] array entry for the share remains active and the client races to auto-reconnect to the share before a second message to reload the smb.conf file can be sent. This has to be done as part of the close-share message processing, as removing the share from the smb.conf file first, then telling the smbd to reload followed by the forcible disconnect message doesn't work as in this sequence of events when the reload message is received the client is still connected to the share, so the ServicePtrs[] entry is still left active. The forcible-disconnect + service reload has to be done together as an atomic operation in order for this to work. Signed-off-by: Jeremy Allison <jra@samba.org> Reviewed-by: David Disseldorp <ddiss@samba.org> Autobuild-User(master): David Disseldorp <ddiss@samba.org> Autobuild-Date(master): Fri Mar 22 20:10:11 CET 2013 on sn-devel-104
149 lines
3.9 KiB
C
149 lines
3.9 KiB
C
/*
|
|
Unix SMB/CIFS implementation.
|
|
Manage connections_struct structures
|
|
Copyright (C) Andrew Tridgell 1998
|
|
Copyright (C) Alexander Bokovoy 2002
|
|
Copyright (C) Jeremy Allison 2010
|
|
|
|
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"
|
|
#include "smbd/smbd.h"
|
|
#include "smbd/globals.h"
|
|
#include "rpc_server/rpc_pipes.h"
|
|
|
|
/****************************************************************************
|
|
Update last used timestamps.
|
|
****************************************************************************/
|
|
|
|
static void conn_lastused_update(struct smbd_server_connection *sconn,time_t t)
|
|
{
|
|
struct connection_struct *conn;
|
|
|
|
for (conn=sconn->connections; conn; conn=conn->next) {
|
|
/* Update if connection wasn't idle. */
|
|
if (conn->lastused != conn->lastused_count) {
|
|
conn->lastused = t;
|
|
conn->lastused_count = t;
|
|
}
|
|
}
|
|
}
|
|
|
|
/****************************************************************************
|
|
Idle inactive connections.
|
|
****************************************************************************/
|
|
|
|
bool conn_idle_all(struct smbd_server_connection *sconn, time_t t)
|
|
{
|
|
int deadtime = lp_deadtime()*60;
|
|
struct connection_struct *conn;
|
|
|
|
conn_lastused_update(sconn, t);
|
|
|
|
if (deadtime <= 0) {
|
|
deadtime = DEFAULT_SMBD_TIMEOUT;
|
|
}
|
|
|
|
for (conn=sconn->connections;conn;conn=conn->next) {
|
|
time_t age = t - conn->lastused;
|
|
|
|
/* close dirptrs on connections that are idle */
|
|
if (age > DPTR_IDLE_TIMEOUT) {
|
|
dptr_idlecnum(conn);
|
|
}
|
|
|
|
if (conn->num_files_open > 0 || age < deadtime) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Check all pipes for any open handles. We cannot
|
|
* idle with a handle open.
|
|
*/
|
|
if (check_open_pipes()) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/****************************************************************************
|
|
Forcibly unmount a share.
|
|
All instances of the parameter 'sharename' share are unmounted.
|
|
The special sharename '*' forces unmount of all shares.
|
|
****************************************************************************/
|
|
|
|
void conn_force_tdis(struct smbd_server_connection *sconn, const char *sharename)
|
|
{
|
|
connection_struct *conn, *next;
|
|
bool close_all = false;
|
|
|
|
if (strcmp(sharename, "*") == 0) {
|
|
close_all = true;
|
|
DEBUG(1, ("conn_force_tdis: Forcing close of all shares\n"));
|
|
}
|
|
|
|
/* SMB1 and SMB 2*/
|
|
for (conn = sconn->connections; conn; conn = next) {
|
|
struct smbXsrv_tcon *tcon;
|
|
bool do_close = false;
|
|
NTSTATUS status;
|
|
uint64_t vuid = UID_FIELD_INVALID;
|
|
|
|
next = conn->next;
|
|
|
|
if (conn->tcon == NULL) {
|
|
continue;
|
|
}
|
|
tcon = conn->tcon;
|
|
|
|
if (close_all) {
|
|
do_close = true;
|
|
} else if (strequal(lp_servicename(talloc_tos(), SNUM(conn)),
|
|
sharename)) {
|
|
DEBUG(1, ("conn_force_tdis: Forcing close of "
|
|
"share '%s' (wire_id=0x%08x)\n",
|
|
tcon->global->share_name,
|
|
tcon->global->tcon_wire_id));
|
|
do_close = true;
|
|
}
|
|
|
|
if (!do_close) {
|
|
continue;
|
|
}
|
|
|
|
if (sconn->using_smb2) {
|
|
vuid = conn->vuid;
|
|
}
|
|
|
|
conn = NULL;
|
|
status = smbXsrv_tcon_disconnect(tcon, vuid);
|
|
if (!NT_STATUS_IS_OK(status)) {
|
|
DEBUG(0, ("conn_force_tdis: "
|
|
"smbXsrv_tcon_disconnect() of share '%s' "
|
|
"(wire_id=0x%08x) failed: %s\n",
|
|
tcon->global->share_name,
|
|
tcon->global->tcon_wire_id,
|
|
nt_errstr(status)));
|
|
}
|
|
|
|
TALLOC_FREE(tcon);
|
|
}
|
|
|
|
change_to_root_user();
|
|
reload_services(sconn, conn_snum_used, true);
|
|
}
|