diff options
Diffstat (limited to 'fs/nfs/symlink.c')
| -rw-r--r-- | fs/nfs/symlink.c | 65 |
1 files changed, 32 insertions, 33 deletions
diff --git a/fs/nfs/symlink.c b/fs/nfs/symlink.c index 05c9e02f4153..58146e935402 100644 --- a/fs/nfs/symlink.c +++ b/fs/nfs/symlink.c @@ -1,3 +1,4 @@ +// SPDX-License-Identifier: GPL-2.0 /* * linux/fs/nfs/symlink.c * @@ -20,59 +21,57 @@ #include <linux/stat.h> #include <linux/mm.h> #include <linux/string.h> -#include <linux/namei.h> /* Symlink caching in the page cache is even more simplistic * and straight-forward than readdir caching. */ -static int nfs_symlink_filler(struct inode *inode, struct page *page) +static int nfs_symlink_filler(struct file *file, struct folio *folio) { + struct inode *inode = folio->mapping->host; int error; - error = NFS_PROTO(inode)->readlink(inode, page, 0, PAGE_SIZE); - if (error < 0) - goto error; - SetPageUptodate(page); - unlock_page(page); - return 0; - -error: - SetPageError(page); - unlock_page(page); - return -EIO; + error = NFS_PROTO(inode)->readlink(inode, &folio->page, 0, PAGE_SIZE); + folio_end_read(folio, error == 0); + return error; } -static void *nfs_follow_link(struct dentry *dentry, struct nameidata *nd) +static const char *nfs_get_link(struct dentry *dentry, + struct inode *inode, + struct delayed_call *done) { - struct inode *inode = dentry->d_inode; - struct page *page; + struct folio *folio; void *err; - err = ERR_PTR(nfs_revalidate_mapping(inode, inode->i_mapping)); - if (err) - goto read_failed; - page = read_cache_page(&inode->i_data, 0, - (filler_t *)nfs_symlink_filler, inode); - if (IS_ERR(page)) { - err = page; - goto read_failed; + if (!dentry) { + err = ERR_PTR(nfs_revalidate_mapping_rcu(inode)); + if (err) + return err; + folio = filemap_get_folio(inode->i_mapping, 0); + if (IS_ERR(folio)) + return ERR_PTR(-ECHILD); + if (!folio_test_uptodate(folio)) { + folio_put(folio); + return ERR_PTR(-ECHILD); + } + } else { + err = ERR_PTR(nfs_revalidate_mapping(inode, inode->i_mapping)); + if (err) + return err; + folio = read_cache_folio(&inode->i_data, 0, nfs_symlink_filler, + NULL); + if (IS_ERR(folio)) + return ERR_CAST(folio); } - nd_set_link(nd, kmap(page)); - return page; - -read_failed: - nd_set_link(nd, err); - return NULL; + set_delayed_call(done, page_put_link, folio); + return folio_address(folio); } /* * symlinks can't do much... */ const struct inode_operations nfs_symlink_inode_operations = { - .readlink = generic_readlink, - .follow_link = nfs_follow_link, - .put_link = page_put_link, + .get_link = nfs_get_link, .getattr = nfs_getattr, .setattr = nfs_setattr, }; |
