On Wed, Oct 25, 2017 at 1:37 PM, Trond Myklebust <trond.myklebust@xxxxxxxxxxxxxxx> wrote: > Ben Coddington has noted the following race between OPEN and CLOSE > on a single client. > > Process 1 Process 2 Server > ========= ========= ====== > > 1) OPEN file > 2) OPEN file > 3) Process OPEN (1) seqid=1 > 4) Process OPEN (2) seqid=2 > 5) Reply OPEN (2) > 6) Receive reply (2) > 7) new stateid, seqid=2 > > 8) CLOSE file, using > stateid w/ seqid=2 > 9) Reply OPEN (1) > 10( Process CLOSE (8) > 11) Reply CLOSE (8) > 12) Forget stateid > file closed > > 13) Receive reply (7) > 14) Forget stateid > file closed. > > 15) Receive reply (1). > 16) New stateid seqid=1 > is really the same > stateid that was > closed. > > IOW: the reply to the first OPEN is delayed. Since "Process 2" does > not wait before closing the file, and it does not cache the closed > stateid, then when the delayed reply is finally received, it is treated > as setting up a new stateid by the client. > > The fix is to ensure that the client processes the OPEN and CLOSE calls > in the same order in which the server processed them. > > This commit ensures that we examine the seqid of the stateid > returned by OPEN. If it is a new stateid, we assume the seqid > must be equal to the value 1, and that each state transition > increments the seqid value by 1 (See RFC7530, Section 9.1.4.2, > and RFC5661, Section 8.2.2). > > If the tracker sees that an OPEN returns with a seqid that is greater > than the cached seqid + 1, then it bumps a flag to ensure that the > caller waits for the RPCs carrying the missing seqids to complete. Please help me with my confusion: I believe the code used to serialize OPENs on the open owner. Then we allowed parallel opens. Without parallel opens this wouldn't have happened, is that correct? Also is your solution again serializing the opens since it says the caller waits for the missing seqid to complete. I read this as the 2nd open waits of the 1st open to complete before proceeding or is that wrong? > Note that there can still be pathologies where the server crashes before > it can even send us the missing seqids. Since the OPEN call is still > holding a slot when it waits here, that could cause the recovery to > stall forever. To avoid that, we time out after a 5 second wait. > > Reported-by: Benjamin Coddington <bcodding@xxxxxxxxxx> > Signed-off-by: Trond Myklebust <trond.myklebust@xxxxxxxxxxxxxxx> > --- > fs/nfs/nfs4_fs.h | 1 + > fs/nfs/nfs4proc.c | 87 +++++++++++++++++++++++++++++++++++++++++++++++-------- > 2 files changed, 76 insertions(+), 12 deletions(-) > > diff --git a/fs/nfs/nfs4_fs.h b/fs/nfs/nfs4_fs.h > index b547d935aaf0..46aeaa2ee700 100644 > --- a/fs/nfs/nfs4_fs.h > +++ b/fs/nfs/nfs4_fs.h > @@ -161,6 +161,7 @@ enum { > NFS_STATE_POSIX_LOCKS, /* Posix locks are supported */ > NFS_STATE_RECOVERY_FAILED, /* OPEN stateid state recovery failed */ > NFS_STATE_MAY_NOTIFY_LOCK, /* server may CB_NOTIFY_LOCK */ > + NFS_STATE_CHANGE_WAIT, /* A state changing operation is outstanding */ > }; > > struct nfs4_state { > diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c > index 96b2077e691d..b426606ef2c4 100644 > --- a/fs/nfs/nfs4proc.c > +++ b/fs/nfs/nfs4proc.c > @@ -1378,6 +1378,28 @@ static bool nfs_open_stateid_recover_openmode(struct nfs4_state *state) > } > #endif /* CONFIG_NFS_V4_1 */ > > +static void nfs_state_log_update_open_stateid(struct nfs4_state *state) > +{ > + if (test_and_clear_bit(NFS_STATE_CHANGE_WAIT, &state->flags)) > + wake_up_bit(&state->flags, NFS_STATE_CHANGE_WAIT); > +} > + > +static void nfs_state_reset_open_stateid(struct nfs4_state *state) > +{ > + state->open_stateid.seqid = 0; > + nfs_state_log_update_open_stateid(state); > +} > + > +static void nfs_state_log_out_of_order_open_stateid(struct nfs4_state *state, > + const nfs4_stateid *stateid) > +{ > + u32 state_seqid = be32_to_cpu(state->open_stateid.seqid); > + u32 stateid_seqid = be32_to_cpu(stateid->seqid); > + > + if (stateid_seqid != state_seqid + 1U) > + set_bit(NFS_STATE_CHANGE_WAIT, &state->flags); > +} > + > static void nfs_test_and_clear_all_open_stateid(struct nfs4_state *state) > { > struct nfs_client *clp = state->owner->so_server->nfs_client; > @@ -1393,18 +1415,34 @@ static void nfs_test_and_clear_all_open_stateid(struct nfs4_state *state) > nfs4_state_mark_reclaim_nograce(clp, state); > } > > +/* > + * Check for whether or not the caller may update the open stateid > + * to the value passed in by stateid. > + * > + * Note: This function relies heavily on the server implementing > + * RFC7530 Section 9.1.4.2, and RFC5661 Section 8.2.2 > + * correctly. > + * i.e. The stateid seqids have to be initialised to 1, and > + * are then incremented on every state transition. > + */ > static bool nfs_need_update_open_stateid(struct nfs4_state *state, > const nfs4_stateid *stateid, nfs4_stateid *freeme) > { > - if (test_and_set_bit(NFS_OPEN_STATE, &state->flags) == 0) > - return true; > - if (!nfs4_stateid_match_other(stateid, &state->open_stateid)) { > - nfs4_stateid_copy(freeme, &state->open_stateid); > - nfs_test_and_clear_all_open_stateid(state); > + if (test_and_set_bit(NFS_OPEN_STATE, &state->flags) == 0) { > + nfs_state_reset_open_stateid(state); > + } else if (!nfs4_stateid_match_other(stateid, &state->open_stateid)) { > + if (stateid->seqid == cpu_to_be32(1)) { > + nfs4_stateid_copy(freeme, &state->open_stateid); > + nfs_test_and_clear_all_open_stateid(state); > + nfs_state_reset_open_stateid(state); > + } else > + set_bit(NFS_STATE_CHANGE_WAIT, &state->flags); > return true; > } > - if (nfs4_stateid_is_newer(stateid, &state->open_stateid)) > + if (nfs4_stateid_is_newer(stateid, &state->open_stateid)) { > + nfs_state_log_out_of_order_open_stateid(state, stateid); > return true; > + } > return false; > } > > @@ -1439,15 +1477,19 @@ static void nfs_clear_open_stateid_locked(struct nfs4_state *state, > } > if (stateid == NULL) > return; > + /* Handle reboot/state expire races */ > + if (!nfs4_stateid_match_other(stateid, &state->open_stateid)) > + return; > /* Handle OPEN+OPEN_DOWNGRADE races */ > - if (nfs4_stateid_match_other(stateid, &state->open_stateid) && > - !nfs4_stateid_is_newer(stateid, &state->open_stateid)) { > + if (!nfs4_stateid_is_newer(stateid, &state->open_stateid)) { > nfs_resync_open_stateid_locked(state); > - return; > + goto out; > } > if (test_bit(NFS_DELEGATED_STATE, &state->flags) == 0) > nfs4_stateid_copy(&state->stateid, stateid); > nfs4_stateid_copy(&state->open_stateid, stateid); > +out: > + nfs_state_log_update_open_stateid(state); > } > > static void nfs_clear_open_stateid(struct nfs4_state *state, > @@ -1467,7 +1509,10 @@ static void nfs_set_open_stateid_locked(struct nfs4_state *state, > const nfs4_stateid *stateid, fmode_t fmode, > nfs4_stateid *freeme) > { > - switch (fmode) { > + int status = 0; > + for (;;) { > + > + switch (fmode) { > case FMODE_READ: > set_bit(NFS_O_RDONLY_STATE, &state->flags); > break; > @@ -1476,12 +1521,30 @@ static void nfs_set_open_stateid_locked(struct nfs4_state *state, > break; > case FMODE_READ|FMODE_WRITE: > set_bit(NFS_O_RDWR_STATE, &state->flags); > + } > + if (!nfs_need_update_open_stateid(state, stateid, freeme)) > + return; > + if (!test_bit(NFS_STATE_CHANGE_WAIT, &state->flags)) > + break; > + if (status) > + break; > + /* > + * Ensure we process the state changes in the same order > + * in which the server processed them by delaying the > + * update of the stateid until we are in sequence. > + */ > + write_sequnlock(&state->seqlock); > + spin_unlock(&state->owner->so_lock); > + status = wait_on_bit_timeout(&state->flags, > + NFS_STATE_CHANGE_WAIT, > + TASK_KILLABLE, 5*HZ); > + spin_lock(&state->owner->so_lock); > + write_seqlock(&state->seqlock); > } > - if (!nfs_need_update_open_stateid(state, stateid, freeme)) > - return; > if (test_bit(NFS_DELEGATED_STATE, &state->flags) == 0) > nfs4_stateid_copy(&state->stateid, stateid); > nfs4_stateid_copy(&state->open_stateid, stateid); > + nfs_state_log_update_open_stateid(state); > } > > static void __update_open_stateid(struct nfs4_state *state, > -- > 2.13.6 > > -- > To unsubscribe from this list: send the line "unsubscribe linux-nfs" in > the body of a message to majordomo@xxxxxxxxxxxxxxx > More majordomo info at http://vger.kernel.org/majordomo-info.html -- To unsubscribe from this list: send the line "unsubscribe linux-nfs" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html