Commit 35dc1d74 authored by Trond Myklebust's avatar Trond Myklebust

NFSv41: Fix up some bugs in the NFS4CLNT_SESSION_DRAINING code

Signed-off-by: default avatarTrond Myklebust <Trond.Myklebust@netapp.com>
parent d61e612a
...@@ -318,13 +318,14 @@ static void renew_lease(const struct nfs_server *server, unsigned long timestamp ...@@ -318,13 +318,14 @@ static void renew_lease(const struct nfs_server *server, unsigned long timestamp
* so we need to scan down from highest_used_slotid to 0 looking for the now * so we need to scan down from highest_used_slotid to 0 looking for the now
* highest slotid in use. * highest slotid in use.
* If none found, highest_used_slotid is set to -1. * If none found, highest_used_slotid is set to -1.
*
* Must be called while holding tbl->slot_tbl_lock
*/ */
static void static void
nfs4_free_slot(struct nfs4_slot_table *tbl, u8 free_slotid) nfs4_free_slot(struct nfs4_slot_table *tbl, u8 free_slotid)
{ {
int slotid = free_slotid; int slotid = free_slotid;
spin_lock(&tbl->slot_tbl_lock);
/* clear used bit in bitmap */ /* clear used bit in bitmap */
__clear_bit(slotid, tbl->used_slots); __clear_bit(slotid, tbl->used_slots);
...@@ -336,7 +337,6 @@ nfs4_free_slot(struct nfs4_slot_table *tbl, u8 free_slotid) ...@@ -336,7 +337,6 @@ nfs4_free_slot(struct nfs4_slot_table *tbl, u8 free_slotid)
else else
tbl->highest_used_slotid = -1; tbl->highest_used_slotid = -1;
} }
spin_unlock(&tbl->slot_tbl_lock);
dprintk("%s: free_slotid %u highest_used_slotid %d\n", __func__, dprintk("%s: free_slotid %u highest_used_slotid %d\n", __func__,
free_slotid, tbl->highest_used_slotid); free_slotid, tbl->highest_used_slotid);
} }
...@@ -351,22 +351,23 @@ static void nfs41_sequence_free_slot(const struct nfs_client *clp, ...@@ -351,22 +351,23 @@ static void nfs41_sequence_free_slot(const struct nfs_client *clp,
/* just wake up the next guy waiting since /* just wake up the next guy waiting since
* we may have not consumed a slot after all */ * we may have not consumed a slot after all */
dprintk("%s: No slot\n", __func__); dprintk("%s: No slot\n", __func__);
} else { return;
nfs4_free_slot(tbl, res->sr_slotid);
res->sr_slotid = NFS4_MAX_SLOT_TABLE;
} }
spin_lock(&tbl->slot_tbl_lock);
nfs4_free_slot(tbl, res->sr_slotid);
/* Signal state manager thread if session is drained */ /* Signal state manager thread if session is drained */
if (test_bit(NFS4CLNT_SESSION_DRAINING, &clp->cl_state)) { if (test_bit(NFS4CLNT_SESSION_DRAINING, &clp->cl_state)) {
spin_lock(&tbl->slot_tbl_lock);
if (tbl->highest_used_slotid == -1) { if (tbl->highest_used_slotid == -1) {
dprintk("%s COMPLETE: Session Drained\n", __func__); dprintk("%s COMPLETE: Session Drained\n", __func__);
complete(&clp->cl_session->complete); complete(&clp->cl_session->complete);
} }
spin_unlock(&tbl->slot_tbl_lock);
} else { } else {
rpc_wake_up_next(&tbl->slot_tbl_waitq); rpc_wake_up_next(&tbl->slot_tbl_waitq);
} }
spin_unlock(&tbl->slot_tbl_lock);
res->sr_slotid = NFS4_MAX_SLOT_TABLE;
} }
static void nfs41_sequence_done(struct nfs_client *clp, static void nfs41_sequence_done(struct nfs_client *clp,
...@@ -470,7 +471,6 @@ static int nfs41_setup_sequence(struct nfs4_session *session, ...@@ -470,7 +471,6 @@ static int nfs41_setup_sequence(struct nfs4_session *session,
*/ */
dprintk("%s Schedule Session Reset\n", __func__); dprintk("%s Schedule Session Reset\n", __func__);
rpc_sleep_on(&tbl->slot_tbl_waitq, task, NULL); rpc_sleep_on(&tbl->slot_tbl_waitq, task, NULL);
nfs4_schedule_state_manager(session->clp);
spin_unlock(&tbl->slot_tbl_lock); spin_unlock(&tbl->slot_tbl_lock);
return -EAGAIN; return -EAGAIN;
} }
...@@ -4489,7 +4489,6 @@ static int nfs4_reset_slot_tables(struct nfs4_session *session) ...@@ -4489,7 +4489,6 @@ static int nfs4_reset_slot_tables(struct nfs4_session *session)
1); 1);
if (status) if (status)
return status; return status;
init_completion(&session->complete);
status = nfs4_reset_slot_table(&session->bc_slot_table, status = nfs4_reset_slot_table(&session->bc_slot_table,
session->bc_attrs.max_reqs, session->bc_attrs.max_reqs,
......
...@@ -1220,10 +1220,10 @@ static int nfs4_reset_session(struct nfs_client *clp) ...@@ -1220,10 +1220,10 @@ static int nfs4_reset_session(struct nfs_client *clp)
struct nfs4_slot_table *tbl = &ses->fc_slot_table; struct nfs4_slot_table *tbl = &ses->fc_slot_table;
int status; int status;
INIT_COMPLETION(ses->complete);
spin_lock(&tbl->slot_tbl_lock); spin_lock(&tbl->slot_tbl_lock);
set_bit(NFS4CLNT_SESSION_DRAINING, &clp->cl_state); set_bit(NFS4CLNT_SESSION_DRAINING, &clp->cl_state);
if (tbl->highest_used_slotid != -1) { if (tbl->highest_used_slotid != -1) {
INIT_COMPLETION(ses->complete);
spin_unlock(&tbl->slot_tbl_lock); spin_unlock(&tbl->slot_tbl_lock);
status = wait_for_completion_interruptible(&ses->complete); status = wait_for_completion_interruptible(&ses->complete);
if (status) /* -ERESTARTSYS */ if (status) /* -ERESTARTSYS */
...@@ -1247,7 +1247,7 @@ static int nfs4_reset_session(struct nfs_client *clp) ...@@ -1247,7 +1247,7 @@ static int nfs4_reset_session(struct nfs_client *clp)
out: out:
/* Wake up the next rpc task even on error */ /* Wake up the next rpc task even on error */
clear_bit(NFS4CLNT_SESSION_DRAINING, &clp->cl_state); clear_bit(NFS4CLNT_SESSION_DRAINING, &clp->cl_state);
rpc_wake_up_next(&clp->cl_session->fc_slot_table.slot_tbl_waitq); rpc_wake_up(&clp->cl_session->fc_slot_table.slot_tbl_waitq);
return status; return status;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment