diff options
author | Shyam Prasad N <sprasad@microsoft.com> | 2024-02-01 11:15:28 +0000 |
---|---|---|
committer | Steve French <stfrench@microsoft.com> | 2024-02-01 12:12:17 -0600 |
commit | 88675b22d34e6e815ad4bde09c590ccb2d50c59d (patch) | |
tree | a189fbadbbeaed3abca0f5f750e0fa43dfcf2a15 | |
parent | e77e15fa5eb1c830597c5ca53ea7af973bae2f78 (diff) | |
download | linux-stable-88675b22d34e6e815ad4bde09c590ccb2d50c59d.tar.gz linux-stable-88675b22d34e6e815ad4bde09c590ccb2d50c59d.tar.bz2 linux-stable-88675b22d34e6e815ad4bde09c590ccb2d50c59d.zip |
cifs: do not search for channel if server is terminating
In order to scale down the channels, the following sequence
of operations happen:
1. server struct is marked for terminate
2. the channel is deallocated in the ses->chans array
3. at a later point the cifsd thread actually terminates the server
Between 2 and 3, there can be calls to find the channel for
a server struct. When that happens, there can be an ugly warning
that's logged. But this is expected.
So this change does two things:
1. in cifs_ses_get_chan_index, if server->terminate is set, return
2. always make sure server->terminate is set with chan_lock held
Signed-off-by: Shyam Prasad N <sprasad@microsoft.com>
Signed-off-by: Steve French <stfrench@microsoft.com>
-rw-r--r-- | fs/smb/client/sess.c | 4 | ||||
-rw-r--r-- | fs/smb/client/smb2pdu.c | 2 |
2 files changed, 5 insertions, 1 deletions
diff --git a/fs/smb/client/sess.c b/fs/smb/client/sess.c index cde81042bebd..3d2548c35c9d 100644 --- a/fs/smb/client/sess.c +++ b/fs/smb/client/sess.c @@ -75,6 +75,10 @@ cifs_ses_get_chan_index(struct cifs_ses *ses, { unsigned int i; + /* if the channel is waiting for termination */ + if (server->terminate) + return CIFS_INVAL_CHAN_INDEX; + for (i = 0; i < ses->chan_count; i++) { if (ses->chans[i].server == server) return i; diff --git a/fs/smb/client/smb2pdu.c b/fs/smb/client/smb2pdu.c index e257b10bc820..c58fa44dd6b0 100644 --- a/fs/smb/client/smb2pdu.c +++ b/fs/smb/client/smb2pdu.c @@ -178,6 +178,7 @@ cifs_chan_skip_or_disable(struct cifs_ses *ses, } ses->chans[chan_index].server = NULL; + server->terminate = true; spin_unlock(&ses->chan_lock); /* @@ -188,7 +189,6 @@ cifs_chan_skip_or_disable(struct cifs_ses *ses, */ cifs_put_tcp_session(server, from_reconnect); - server->terminate = true; cifs_signal_cifsd_for_reconnect(server, false); /* mark primary server as needing reconnect */ |