|  | /* SPDX-License-Identifier: GPL-2.0-or-later */ | 
|  | /* NFS filesystem cache interface definitions | 
|  | * | 
|  | * Copyright (C) 2008 Red Hat, Inc. All Rights Reserved. | 
|  | * Written by David Howells (dhowells@redhat.com) | 
|  | */ | 
|  |  | 
|  | #ifndef _NFS_FSCACHE_H | 
|  | #define _NFS_FSCACHE_H | 
|  |  | 
|  | #include <linux/swap.h> | 
|  | #include <linux/nfs_fs.h> | 
|  | #include <linux/nfs_mount.h> | 
|  | #include <linux/nfs4_mount.h> | 
|  | #include <linux/fscache.h> | 
|  | #include <linux/iversion.h> | 
|  |  | 
|  | #ifdef CONFIG_NFS_FSCACHE | 
|  |  | 
|  | /* | 
|  | * Definition of the auxiliary data attached to NFS inode storage objects | 
|  | * within the cache. | 
|  | * | 
|  | * The contents of this struct are recorded in the on-disk local cache in the | 
|  | * auxiliary data attached to the data storage object backing an inode.  This | 
|  | * permits coherency to be managed when a new inode binds to an already extant | 
|  | * cache object. | 
|  | */ | 
|  | struct nfs_fscache_inode_auxdata { | 
|  | s64	mtime_sec; | 
|  | s64	mtime_nsec; | 
|  | s64	ctime_sec; | 
|  | s64	ctime_nsec; | 
|  | u64	change_attr; | 
|  | }; | 
|  |  | 
|  | struct nfs_netfs_io_data { | 
|  | /* | 
|  | * NFS may split a netfs_io_subrequest into multiple RPCs, each | 
|  | * with their own read completion.  In netfs, we can only call | 
|  | * netfs_subreq_terminated() once for each subrequest.  Use the | 
|  | * refcount here to double as a marker of the last RPC completion, | 
|  | * and only call netfs via netfs_subreq_terminated() once. | 
|  | */ | 
|  | refcount_t			refcount; | 
|  | struct netfs_io_subrequest	*sreq; | 
|  |  | 
|  | /* | 
|  | * Final disposition of the netfs_io_subrequest, sent in | 
|  | * netfs_subreq_terminated() | 
|  | */ | 
|  | atomic64_t	transferred; | 
|  | int		error; | 
|  | }; | 
|  |  | 
|  | static inline void nfs_netfs_get(struct nfs_netfs_io_data *netfs) | 
|  | { | 
|  | refcount_inc(&netfs->refcount); | 
|  | } | 
|  |  | 
|  | static inline void nfs_netfs_put(struct nfs_netfs_io_data *netfs) | 
|  | { | 
|  | ssize_t final_len; | 
|  |  | 
|  | /* Only the last RPC completion should call netfs_subreq_terminated() */ | 
|  | if (!refcount_dec_and_test(&netfs->refcount)) | 
|  | return; | 
|  |  | 
|  | /* | 
|  | * The NFS pageio interface may read a complete page, even when netfs | 
|  | * only asked for a partial page.  Specifically, this may be seen when | 
|  | * one thread is truncating a file while another one is reading the last | 
|  | * page of the file. | 
|  | * Correct the final length here to be no larger than the netfs subrequest | 
|  | * length, and thus avoid netfs's "Subreq overread" warning message. | 
|  | */ | 
|  | final_len = min_t(s64, netfs->sreq->len, atomic64_read(&netfs->transferred)); | 
|  | netfs_subreq_terminated(netfs->sreq, netfs->error ?: final_len, false); | 
|  | kfree(netfs); | 
|  | } | 
|  | static inline void nfs_netfs_inode_init(struct nfs_inode *nfsi) | 
|  | { | 
|  | netfs_inode_init(&nfsi->netfs, &nfs_netfs_ops); | 
|  | } | 
|  | extern void nfs_netfs_initiate_read(struct nfs_pgio_header *hdr); | 
|  | extern void nfs_netfs_read_completion(struct nfs_pgio_header *hdr); | 
|  | extern int nfs_netfs_folio_unlock(struct folio *folio); | 
|  |  | 
|  | /* | 
|  | * fscache.c | 
|  | */ | 
|  | extern int nfs_fscache_get_super_cookie(struct super_block *, const char *, int); | 
|  | extern void nfs_fscache_release_super_cookie(struct super_block *); | 
|  |  | 
|  | extern void nfs_fscache_init_inode(struct inode *); | 
|  | extern void nfs_fscache_clear_inode(struct inode *); | 
|  | extern void nfs_fscache_open_file(struct inode *, struct file *); | 
|  | extern void nfs_fscache_release_file(struct inode *, struct file *); | 
|  | extern int nfs_netfs_readahead(struct readahead_control *ractl); | 
|  | extern int nfs_netfs_read_folio(struct file *file, struct folio *folio); | 
|  |  | 
|  | static inline bool nfs_fscache_release_folio(struct folio *folio, gfp_t gfp) | 
|  | { | 
|  | if (folio_test_fscache(folio)) { | 
|  | if (current_is_kswapd() || !(gfp & __GFP_FS)) | 
|  | return false; | 
|  | folio_wait_fscache(folio); | 
|  | } | 
|  | fscache_note_page_release(netfs_i_cookie(netfs_inode(folio->mapping->host))); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static inline void nfs_fscache_update_auxdata(struct nfs_fscache_inode_auxdata *auxdata, | 
|  | struct inode *inode) | 
|  | { | 
|  | memset(auxdata, 0, sizeof(*auxdata)); | 
|  | auxdata->mtime_sec  = inode->i_mtime.tv_sec; | 
|  | auxdata->mtime_nsec = inode->i_mtime.tv_nsec; | 
|  | auxdata->ctime_sec  = inode_get_ctime(inode).tv_sec; | 
|  | auxdata->ctime_nsec = inode_get_ctime(inode).tv_nsec; | 
|  |  | 
|  | if (NFS_SERVER(inode)->nfs_client->rpc_ops->version == 4) | 
|  | auxdata->change_attr = inode_peek_iversion_raw(inode); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Invalidate the contents of fscache for this inode.  This will not sleep. | 
|  | */ | 
|  | static inline void nfs_fscache_invalidate(struct inode *inode, int flags) | 
|  | { | 
|  | struct nfs_fscache_inode_auxdata auxdata; | 
|  | struct fscache_cookie *cookie =  netfs_i_cookie(&NFS_I(inode)->netfs); | 
|  |  | 
|  | nfs_fscache_update_auxdata(&auxdata, inode); | 
|  | fscache_invalidate(cookie, &auxdata, i_size_read(inode), flags); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * indicate the client caching state as readable text | 
|  | */ | 
|  | static inline const char *nfs_server_fscache_state(struct nfs_server *server) | 
|  | { | 
|  | if (server->fscache) | 
|  | return "yes"; | 
|  | return "no "; | 
|  | } | 
|  |  | 
|  | static inline void nfs_netfs_set_pgio_header(struct nfs_pgio_header *hdr, | 
|  | struct nfs_pageio_descriptor *desc) | 
|  | { | 
|  | hdr->netfs = desc->pg_netfs; | 
|  | } | 
|  | static inline void nfs_netfs_set_pageio_descriptor(struct nfs_pageio_descriptor *desc, | 
|  | struct nfs_pgio_header *hdr) | 
|  | { | 
|  | desc->pg_netfs = hdr->netfs; | 
|  | } | 
|  | static inline void nfs_netfs_reset_pageio_descriptor(struct nfs_pageio_descriptor *desc) | 
|  | { | 
|  | desc->pg_netfs = NULL; | 
|  | } | 
|  | #else /* CONFIG_NFS_FSCACHE */ | 
|  | static inline void nfs_netfs_inode_init(struct nfs_inode *nfsi) {} | 
|  | static inline void nfs_netfs_initiate_read(struct nfs_pgio_header *hdr) {} | 
|  | static inline void nfs_netfs_read_completion(struct nfs_pgio_header *hdr) {} | 
|  | static inline int nfs_netfs_folio_unlock(struct folio *folio) | 
|  | { | 
|  | return 1; | 
|  | } | 
|  | static inline void nfs_fscache_release_super_cookie(struct super_block *sb) {} | 
|  |  | 
|  | static inline void nfs_fscache_init_inode(struct inode *inode) {} | 
|  | static inline void nfs_fscache_clear_inode(struct inode *inode) {} | 
|  | static inline void nfs_fscache_open_file(struct inode *inode, | 
|  | struct file *filp) {} | 
|  | static inline void nfs_fscache_release_file(struct inode *inode, struct file *file) {} | 
|  | static inline int nfs_netfs_readahead(struct readahead_control *ractl) | 
|  | { | 
|  | return -ENOBUFS; | 
|  | } | 
|  | static inline int nfs_netfs_read_folio(struct file *file, struct folio *folio) | 
|  | { | 
|  | return -ENOBUFS; | 
|  | } | 
|  |  | 
|  | static inline bool nfs_fscache_release_folio(struct folio *folio, gfp_t gfp) | 
|  | { | 
|  | return true; /* may release folio */ | 
|  | } | 
|  | static inline void nfs_fscache_invalidate(struct inode *inode, int flags) {} | 
|  |  | 
|  | static inline const char *nfs_server_fscache_state(struct nfs_server *server) | 
|  | { | 
|  | return "no "; | 
|  | } | 
|  | static inline void nfs_netfs_set_pgio_header(struct nfs_pgio_header *hdr, | 
|  | struct nfs_pageio_descriptor *desc) {} | 
|  | static inline void nfs_netfs_set_pageio_descriptor(struct nfs_pageio_descriptor *desc, | 
|  | struct nfs_pgio_header *hdr) {} | 
|  | static inline void nfs_netfs_reset_pageio_descriptor(struct nfs_pageio_descriptor *desc) {} | 
|  | #endif /* CONFIG_NFS_FSCACHE */ | 
|  | #endif /* _NFS_FSCACHE_H */ |