162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci *  linux/fs/nfs/symlink.c
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci *  Copyright (C) 1992  Rick Sladkey
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci *  Optimization changes Copyright (C) 1994 Florian La Roche
862306a36Sopenharmony_ci *
962306a36Sopenharmony_ci *  Jun 7 1999, cache symlink lookups in the page cache.  -DaveM
1062306a36Sopenharmony_ci *
1162306a36Sopenharmony_ci *  nfs symlink handling code
1262306a36Sopenharmony_ci */
1362306a36Sopenharmony_ci
1462306a36Sopenharmony_ci#include <linux/time.h>
1562306a36Sopenharmony_ci#include <linux/errno.h>
1662306a36Sopenharmony_ci#include <linux/sunrpc/clnt.h>
1762306a36Sopenharmony_ci#include <linux/nfs.h>
1862306a36Sopenharmony_ci#include <linux/nfs2.h>
1962306a36Sopenharmony_ci#include <linux/nfs_fs.h>
2062306a36Sopenharmony_ci#include <linux/pagemap.h>
2162306a36Sopenharmony_ci#include <linux/stat.h>
2262306a36Sopenharmony_ci#include <linux/mm.h>
2362306a36Sopenharmony_ci#include <linux/string.h>
2462306a36Sopenharmony_ci
2562306a36Sopenharmony_ci/* Symlink caching in the page cache is even more simplistic
2662306a36Sopenharmony_ci * and straight-forward than readdir caching.
2762306a36Sopenharmony_ci */
2862306a36Sopenharmony_ci
2962306a36Sopenharmony_cistatic int nfs_symlink_filler(struct file *file, struct folio *folio)
3062306a36Sopenharmony_ci{
3162306a36Sopenharmony_ci	struct inode *inode = folio->mapping->host;
3262306a36Sopenharmony_ci	int error;
3362306a36Sopenharmony_ci
3462306a36Sopenharmony_ci	error = NFS_PROTO(inode)->readlink(inode, &folio->page, 0, PAGE_SIZE);
3562306a36Sopenharmony_ci	if (error < 0)
3662306a36Sopenharmony_ci		goto error;
3762306a36Sopenharmony_ci	folio_mark_uptodate(folio);
3862306a36Sopenharmony_ci	folio_unlock(folio);
3962306a36Sopenharmony_ci	return 0;
4062306a36Sopenharmony_ci
4162306a36Sopenharmony_cierror:
4262306a36Sopenharmony_ci	folio_set_error(folio);
4362306a36Sopenharmony_ci	folio_unlock(folio);
4462306a36Sopenharmony_ci	return -EIO;
4562306a36Sopenharmony_ci}
4662306a36Sopenharmony_ci
4762306a36Sopenharmony_cistatic const char *nfs_get_link(struct dentry *dentry,
4862306a36Sopenharmony_ci				struct inode *inode,
4962306a36Sopenharmony_ci				struct delayed_call *done)
5062306a36Sopenharmony_ci{
5162306a36Sopenharmony_ci	struct page *page;
5262306a36Sopenharmony_ci	void *err;
5362306a36Sopenharmony_ci
5462306a36Sopenharmony_ci	if (!dentry) {
5562306a36Sopenharmony_ci		err = ERR_PTR(nfs_revalidate_mapping_rcu(inode));
5662306a36Sopenharmony_ci		if (err)
5762306a36Sopenharmony_ci			return err;
5862306a36Sopenharmony_ci		page = find_get_page(inode->i_mapping, 0);
5962306a36Sopenharmony_ci		if (!page)
6062306a36Sopenharmony_ci			return ERR_PTR(-ECHILD);
6162306a36Sopenharmony_ci		if (!PageUptodate(page)) {
6262306a36Sopenharmony_ci			put_page(page);
6362306a36Sopenharmony_ci			return ERR_PTR(-ECHILD);
6462306a36Sopenharmony_ci		}
6562306a36Sopenharmony_ci	} else {
6662306a36Sopenharmony_ci		err = ERR_PTR(nfs_revalidate_mapping(inode, inode->i_mapping));
6762306a36Sopenharmony_ci		if (err)
6862306a36Sopenharmony_ci			return err;
6962306a36Sopenharmony_ci		page = read_cache_page(&inode->i_data, 0, nfs_symlink_filler,
7062306a36Sopenharmony_ci				NULL);
7162306a36Sopenharmony_ci		if (IS_ERR(page))
7262306a36Sopenharmony_ci			return ERR_CAST(page);
7362306a36Sopenharmony_ci	}
7462306a36Sopenharmony_ci	set_delayed_call(done, page_put_link, page);
7562306a36Sopenharmony_ci	return page_address(page);
7662306a36Sopenharmony_ci}
7762306a36Sopenharmony_ci
7862306a36Sopenharmony_ci/*
7962306a36Sopenharmony_ci * symlinks can't do much...
8062306a36Sopenharmony_ci */
8162306a36Sopenharmony_ciconst struct inode_operations nfs_symlink_inode_operations = {
8262306a36Sopenharmony_ci	.get_link	= nfs_get_link,
8362306a36Sopenharmony_ci	.getattr	= nfs_getattr,
8462306a36Sopenharmony_ci	.setattr	= nfs_setattr,
8562306a36Sopenharmony_ci};
86