Commit 078000d0 authored by Trond Myklebust's avatar Trond Myklebust

pNFS: We want return-on-close to complete when evicting the inode

If the inode is being evicted, it should be safe to run return-on-close,
so we should do it to ensure we don't inadvertently leak layout segments.

Fixes: 1c5bd76d ("pNFS: Enable layoutreturn operation for return-on-close")
Signed-off-by: default avatarTrond Myklebust <trond.myklebust@hammerspace.com>
parent 67bbceed
...@@ -3536,10 +3536,8 @@ static void nfs4_close_done(struct rpc_task *task, void *data) ...@@ -3536,10 +3536,8 @@ static void nfs4_close_done(struct rpc_task *task, void *data)
trace_nfs4_close(state, &calldata->arg, &calldata->res, task->tk_status); trace_nfs4_close(state, &calldata->arg, &calldata->res, task->tk_status);
/* Handle Layoutreturn errors */ /* Handle Layoutreturn errors */
if (pnfs_roc_done(task, calldata->inode, if (pnfs_roc_done(task, &calldata->arg.lr_args, &calldata->res.lr_res,
&calldata->arg.lr_args, &calldata->res.lr_ret) == -EAGAIN)
&calldata->res.lr_res,
&calldata->res.lr_ret) == -EAGAIN)
goto out_restart; goto out_restart;
/* hmm. we are done with the inode, and in the process of freeing /* hmm. we are done with the inode, and in the process of freeing
...@@ -6384,10 +6382,8 @@ static void nfs4_delegreturn_done(struct rpc_task *task, void *calldata) ...@@ -6384,10 +6382,8 @@ static void nfs4_delegreturn_done(struct rpc_task *task, void *calldata)
trace_nfs4_delegreturn_exit(&data->args, &data->res, task->tk_status); trace_nfs4_delegreturn_exit(&data->args, &data->res, task->tk_status);
/* Handle Layoutreturn errors */ /* Handle Layoutreturn errors */
if (pnfs_roc_done(task, data->inode, if (pnfs_roc_done(task, &data->args.lr_args, &data->res.lr_res,
&data->args.lr_args, &data->res.lr_ret) == -EAGAIN)
&data->res.lr_res,
&data->res.lr_ret) == -EAGAIN)
goto out_restart; goto out_restart;
switch (task->tk_status) { switch (task->tk_status) {
...@@ -6441,10 +6437,10 @@ static void nfs4_delegreturn_release(void *calldata) ...@@ -6441,10 +6437,10 @@ static void nfs4_delegreturn_release(void *calldata)
struct nfs4_delegreturndata *data = calldata; struct nfs4_delegreturndata *data = calldata;
struct inode *inode = data->inode; struct inode *inode = data->inode;
if (data->lr.roc)
pnfs_roc_release(&data->lr.arg, &data->lr.res,
data->res.lr_ret);
if (inode) { if (inode) {
if (data->lr.roc)
pnfs_roc_release(&data->lr.arg, &data->lr.res,
data->res.lr_ret);
nfs_post_op_update_inode_force_wcc(inode, &data->fattr); nfs_post_op_update_inode_force_wcc(inode, &data->fattr);
nfs_iput_and_deactive(inode); nfs_iput_and_deactive(inode);
} }
...@@ -6520,16 +6516,14 @@ static int _nfs4_proc_delegreturn(struct inode *inode, const struct cred *cred, ...@@ -6520,16 +6516,14 @@ static int _nfs4_proc_delegreturn(struct inode *inode, const struct cred *cred,
nfs_fattr_init(data->res.fattr); nfs_fattr_init(data->res.fattr);
data->timestamp = jiffies; data->timestamp = jiffies;
data->rpc_status = 0; data->rpc_status = 0;
data->lr.roc = pnfs_roc(inode, &data->lr.arg, &data->lr.res, cred);
data->inode = nfs_igrab_and_active(inode); data->inode = nfs_igrab_and_active(inode);
if (data->inode) { if (data->inode || issync) {
data->lr.roc = pnfs_roc(inode, &data->lr.arg, &data->lr.res,
cred);
if (data->lr.roc) { if (data->lr.roc) {
data->args.lr_args = &data->lr.arg; data->args.lr_args = &data->lr.arg;
data->res.lr_res = &data->lr.res; data->res.lr_res = &data->lr.res;
} }
} else if (data->lr.roc) {
pnfs_roc_release(&data->lr.arg, &data->lr.res, 0);
data->lr.roc = false;
} }
task_setup_data.callback_data = data; task_setup_data.callback_data = data;
......
...@@ -1509,10 +1509,8 @@ bool pnfs_roc(struct inode *ino, ...@@ -1509,10 +1509,8 @@ bool pnfs_roc(struct inode *ino,
return false; return false;
} }
int pnfs_roc_done(struct rpc_task *task, struct inode *inode, int pnfs_roc_done(struct rpc_task *task, struct nfs4_layoutreturn_args **argpp,
struct nfs4_layoutreturn_args **argpp, struct nfs4_layoutreturn_res **respp, int *ret)
struct nfs4_layoutreturn_res **respp,
int *ret)
{ {
struct nfs4_layoutreturn_args *arg = *argpp; struct nfs4_layoutreturn_args *arg = *argpp;
int retval = -EAGAIN; int retval = -EAGAIN;
...@@ -1545,7 +1543,7 @@ int pnfs_roc_done(struct rpc_task *task, struct inode *inode, ...@@ -1545,7 +1543,7 @@ int pnfs_roc_done(struct rpc_task *task, struct inode *inode,
return 0; return 0;
case -NFS4ERR_OLD_STATEID: case -NFS4ERR_OLD_STATEID:
if (!nfs4_layout_refresh_old_stateid(&arg->stateid, if (!nfs4_layout_refresh_old_stateid(&arg->stateid,
&arg->range, inode)) &arg->range, arg->inode))
break; break;
*ret = -NFS4ERR_NOMATCHING_LAYOUT; *ret = -NFS4ERR_NOMATCHING_LAYOUT;
return -EAGAIN; return -EAGAIN;
......
...@@ -297,10 +297,8 @@ bool pnfs_roc(struct inode *ino, ...@@ -297,10 +297,8 @@ bool pnfs_roc(struct inode *ino,
struct nfs4_layoutreturn_args *args, struct nfs4_layoutreturn_args *args,
struct nfs4_layoutreturn_res *res, struct nfs4_layoutreturn_res *res,
const struct cred *cred); const struct cred *cred);
int pnfs_roc_done(struct rpc_task *task, struct inode *inode, int pnfs_roc_done(struct rpc_task *task, struct nfs4_layoutreturn_args **argpp,
struct nfs4_layoutreturn_args **argpp, struct nfs4_layoutreturn_res **respp, int *ret);
struct nfs4_layoutreturn_res **respp,
int *ret);
void pnfs_roc_release(struct nfs4_layoutreturn_args *args, void pnfs_roc_release(struct nfs4_layoutreturn_args *args,
struct nfs4_layoutreturn_res *res, struct nfs4_layoutreturn_res *res,
int ret); int ret);
...@@ -772,7 +770,7 @@ pnfs_roc(struct inode *ino, ...@@ -772,7 +770,7 @@ pnfs_roc(struct inode *ino,
} }
static inline int static inline int
pnfs_roc_done(struct rpc_task *task, struct inode *inode, pnfs_roc_done(struct rpc_task *task,
struct nfs4_layoutreturn_args **argpp, struct nfs4_layoutreturn_args **argpp,
struct nfs4_layoutreturn_res **respp, struct nfs4_layoutreturn_res **respp,
int *ret) int *ret)
......
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