aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorTrond Myklebust <trond.myklebust@primarydata.com>2014-08-25 22:33:12 -0400
committerTrond Myklebust <trond.myklebust@primarydata.com>2014-08-26 16:17:48 -0400
commitaee7af356e151494d5014f57b33460b162f181b5 (patch)
treec551c93d552db5b2992043d1c68404f7dd907df1
parent52addcf9d6669fa439387610bc65c92fa0980cef (diff)
NFSv4: Fix problems with close in the presence of a delegation
In the presence of delegations, we can no longer assume that the state->n_rdwr, state->n_rdonly, state->n_wronly reflect the open stateid share mode, and so we need to calculate the initial value for calldata->arg.fmode using the state->flags. Reported-by: James Drews <drews@engr.wisc.edu> Fixes: 88069f77e1ac5 (NFSv41: Fix a potential state leakage when...) Cc: stable@vger.kernel.org # 2.6.33+ Signed-off-by: Trond Myklebust <trond.myklebust@primarydata.com>
-rw-r--r--fs/nfs/nfs4proc.c17
1 files changed, 12 insertions, 5 deletions
diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c
index 75ae8d22f067..ff94a2f36051 100644
--- a/fs/nfs/nfs4proc.c
+++ b/fs/nfs/nfs4proc.c
@@ -2601,6 +2601,7 @@ static void nfs4_close_prepare(struct rpc_task *task, void *data)
2601 struct nfs4_closedata *calldata = data; 2601 struct nfs4_closedata *calldata = data;
2602 struct nfs4_state *state = calldata->state; 2602 struct nfs4_state *state = calldata->state;
2603 struct inode *inode = calldata->inode; 2603 struct inode *inode = calldata->inode;
2604 bool is_rdonly, is_wronly, is_rdwr;
2604 int call_close = 0; 2605 int call_close = 0;
2605 2606
2606 dprintk("%s: begin!\n", __func__); 2607 dprintk("%s: begin!\n", __func__);
@@ -2608,18 +2609,24 @@ static void nfs4_close_prepare(struct rpc_task *task, void *data)
2608 goto out_wait; 2609 goto out_wait;
2609 2610
2610 task->tk_msg.rpc_proc = &nfs4_procedures[NFSPROC4_CLNT_OPEN_DOWNGRADE]; 2611 task->tk_msg.rpc_proc = &nfs4_procedures[NFSPROC4_CLNT_OPEN_DOWNGRADE];
2611 calldata->arg.fmode = FMODE_READ|FMODE_WRITE;
2612 spin_lock(&state->owner->so_lock); 2612 spin_lock(&state->owner->so_lock);
2613 is_rdwr = test_bit(NFS_O_RDWR_STATE, &state->flags);
2614 is_rdonly = test_bit(NFS_O_RDONLY_STATE, &state->flags);
2615 is_wronly = test_bit(NFS_O_WRONLY_STATE, &state->flags);
2616 /* Calculate the current open share mode */
2617 calldata->arg.fmode = 0;
2618 if (is_rdonly || is_rdwr)
2619 calldata->arg.fmode |= FMODE_READ;
2620 if (is_wronly || is_rdwr)
2621 calldata->arg.fmode |= FMODE_WRITE;
2613 /* Calculate the change in open mode */ 2622 /* Calculate the change in open mode */
2614 if (state->n_rdwr == 0) { 2623 if (state->n_rdwr == 0) {
2615 if (state->n_rdonly == 0) { 2624 if (state->n_rdonly == 0) {
2616 call_close |= test_bit(NFS_O_RDONLY_STATE, &state->flags); 2625 call_close |= is_rdonly || is_rdwr;
2617 call_close |= test_bit(NFS_O_RDWR_STATE, &state->flags);
2618 calldata->arg.fmode &= ~FMODE_READ; 2626 calldata->arg.fmode &= ~FMODE_READ;
2619 } 2627 }
2620 if (state->n_wronly == 0) { 2628 if (state->n_wronly == 0) {
2621 call_close |= test_bit(NFS_O_WRONLY_STATE, &state->flags); 2629 call_close |= is_wronly || is_rdwr;
2622 call_close |= test_bit(NFS_O_RDWR_STATE, &state->flags);
2623 calldata->arg.fmode &= ~FMODE_WRITE; 2630 calldata->arg.fmode &= ~FMODE_WRITE;
2624 } 2631 }
2625 } 2632 }