TCON Reconnect during STATUS_NETWORK_NAME_DELETED
When server returns error STATUS_NETWORK_NAME_DELETED, TCON must be marked for reconnect. So, subsequent IO does the tree connect again. Signed-off-by: Rohith Surabattula <rohiths@microsoft.com> Reviewed-by: Pavel Shilovsky <pshilov@microsoft.com> Reviewed-by: Shyam Prasad N <sprasad@microsoft.com> Signed-off-by: Steve French <stfrench@microsoft.com>
This commit is contained in:
parent
23bda5e651
commit
9e550b0852
|
@ -505,6 +505,8 @@ struct smb_version_operations {
|
|||
loff_t (*llseek)(struct file *, struct cifs_tcon *, loff_t, int);
|
||||
/* Check for STATUS_IO_TIMEOUT */
|
||||
bool (*is_status_io_timeout)(char *buf);
|
||||
/* Check for STATUS_NETWORK_NAME_DELETED */
|
||||
void (*is_network_name_deleted)(char *buf, struct TCP_Server_Info *srv);
|
||||
};
|
||||
|
||||
struct smb_version_values {
|
||||
|
|
|
@ -995,6 +995,10 @@ next_pdu:
|
|||
if (mids[i] != NULL) {
|
||||
mids[i]->resp_buf_size = server->pdu_size;
|
||||
|
||||
if (bufs[i] && server->ops->is_network_name_deleted)
|
||||
server->ops->is_network_name_deleted(bufs[i],
|
||||
server);
|
||||
|
||||
if (!mids[i]->multiRsp || mids[i]->multiEnd)
|
||||
mids[i]->callback(mids[i]);
|
||||
|
||||
|
|
|
@ -2451,6 +2451,33 @@ smb2_is_status_io_timeout(char *buf)
|
|||
return false;
|
||||
}
|
||||
|
||||
static void
|
||||
smb2_is_network_name_deleted(char *buf, struct TCP_Server_Info *server)
|
||||
{
|
||||
struct smb2_sync_hdr *shdr = (struct smb2_sync_hdr *)buf;
|
||||
struct list_head *tmp, *tmp1;
|
||||
struct cifs_ses *ses;
|
||||
struct cifs_tcon *tcon;
|
||||
|
||||
if (shdr->Status == STATUS_NETWORK_NAME_DELETED) {
|
||||
spin_lock(&cifs_tcp_ses_lock);
|
||||
list_for_each(tmp, &server->smb_ses_list) {
|
||||
ses = list_entry(tmp, struct cifs_ses, smb_ses_list);
|
||||
list_for_each(tmp1, &ses->tcon_list) {
|
||||
tcon = list_entry(tmp1, struct cifs_tcon, tcon_list);
|
||||
if (tcon->tid == shdr->TreeId) {
|
||||
tcon->need_reconnect = true;
|
||||
spin_unlock(&cifs_tcp_ses_lock);
|
||||
pr_warn_once("Server share %s deleted.\n",
|
||||
tcon->treeName);
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
spin_unlock(&cifs_tcp_ses_lock);
|
||||
}
|
||||
}
|
||||
|
||||
static int
|
||||
smb2_oplock_response(struct cifs_tcon *tcon, struct cifs_fid *fid,
|
||||
struct cifsInodeInfo *cinode)
|
||||
|
@ -4638,6 +4665,10 @@ static void smb2_decrypt_offload(struct work_struct *work)
|
|||
#ifdef CONFIG_CIFS_STATS2
|
||||
mid->when_received = jiffies;
|
||||
#endif
|
||||
if (dw->server->ops->is_network_name_deleted)
|
||||
dw->server->ops->is_network_name_deleted(dw->buf,
|
||||
dw->server);
|
||||
|
||||
mid->callback(mid);
|
||||
} else {
|
||||
spin_lock(&GlobalMid_Lock);
|
||||
|
@ -4756,6 +4787,12 @@ non_offloaded_decrypt:
|
|||
rc = handle_read_data(server, *mid, buf,
|
||||
server->vals->read_rsp_size,
|
||||
pages, npages, len, false);
|
||||
if (rc >= 0) {
|
||||
if (server->ops->is_network_name_deleted) {
|
||||
server->ops->is_network_name_deleted(buf,
|
||||
server);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
free_pages:
|
||||
|
@ -5105,6 +5142,7 @@ struct smb_version_operations smb20_operations = {
|
|||
.fiemap = smb3_fiemap,
|
||||
.llseek = smb3_llseek,
|
||||
.is_status_io_timeout = smb2_is_status_io_timeout,
|
||||
.is_network_name_deleted = smb2_is_network_name_deleted,
|
||||
};
|
||||
|
||||
struct smb_version_operations smb21_operations = {
|
||||
|
@ -5206,6 +5244,7 @@ struct smb_version_operations smb21_operations = {
|
|||
.fiemap = smb3_fiemap,
|
||||
.llseek = smb3_llseek,
|
||||
.is_status_io_timeout = smb2_is_status_io_timeout,
|
||||
.is_network_name_deleted = smb2_is_network_name_deleted,
|
||||
};
|
||||
|
||||
struct smb_version_operations smb30_operations = {
|
||||
|
@ -5319,6 +5358,7 @@ struct smb_version_operations smb30_operations = {
|
|||
.fiemap = smb3_fiemap,
|
||||
.llseek = smb3_llseek,
|
||||
.is_status_io_timeout = smb2_is_status_io_timeout,
|
||||
.is_network_name_deleted = smb2_is_network_name_deleted,
|
||||
};
|
||||
|
||||
struct smb_version_operations smb311_operations = {
|
||||
|
@ -5432,6 +5472,7 @@ struct smb_version_operations smb311_operations = {
|
|||
.fiemap = smb3_fiemap,
|
||||
.llseek = smb3_llseek,
|
||||
.is_status_io_timeout = smb2_is_status_io_timeout,
|
||||
.is_network_name_deleted = smb2_is_network_name_deleted,
|
||||
};
|
||||
|
||||
struct smb_version_values smb20_values = {
|
||||
|
|
Loading…
Reference in New Issue