nfsd: Prevent truncation of an unlinked inode from blocking access to its directory
authorYu Hsiang Huang <nickhuang@synology.com>
Fri, 14 May 2021 03:58:29 +0000 (11:58 +0800)
committerJ. Bruce Fields <bfields@redhat.com>
Tue, 25 May 2021 21:06:51 +0000 (17:06 -0400)
Truncation of an unlinked inode may take a long time for I/O waiting, and
it doesn't have to prevent access to the directory. Thus, let truncation
occur outside the directory's mutex, just like do_unlinkat() does.

Signed-off-by: Yu Hsiang Huang <nickhuang@synology.com>
Signed-off-by: Bing Jing Chang <bingjingc@synology.com>
Signed-off-by: Robbie Ko <robbieko@synology.com>
Signed-off-by: J. Bruce Fields <bfields@redhat.com>
fs/nfsd/vfs.c

index 15adf1f6ab21fc7d258227046f76cef1aad20687..39948f1307123256027d3513738836974f6f2302 100644 (file)
@@ -1859,6 +1859,7 @@ nfsd_unlink(struct svc_rqst *rqstp, struct svc_fh *fhp, int type,
 {
        struct dentry   *dentry, *rdentry;
        struct inode    *dirp;
+       struct inode    *rinode;
        __be32          err;
        int             host_err;
 
@@ -1887,6 +1888,8 @@ nfsd_unlink(struct svc_rqst *rqstp, struct svc_fh *fhp, int type,
                host_err = -ENOENT;
                goto out_drop_write;
        }
+       rinode = d_inode(rdentry);
+       ihold(rinode);
 
        if (!type)
                type = d_inode(rdentry)->i_mode & S_IFMT;
@@ -1902,6 +1905,8 @@ nfsd_unlink(struct svc_rqst *rqstp, struct svc_fh *fhp, int type,
        if (!host_err)
                host_err = commit_metadata(fhp);
        dput(rdentry);
+       fh_unlock(fhp);
+       iput(rinode);    /* truncate the inode here */
 
 out_drop_write:
        fh_drop_write(fhp);