| // SPDX-License-Identifier: GPL-2.0-only |
| /* |
| * V9FS cache definitions. |
| * |
| * Copyright (C) 2009 by Abhishek Kulkarni <adkulkar@umail.iu.edu> |
| */ |
| |
| #include <linux/jiffies.h> |
| #include <linux/file.h> |
| #include <linux/slab.h> |
| #include <linux/stat.h> |
| #include <linux/sched.h> |
| #include <linux/fs.h> |
| #include <net/9p/9p.h> |
| |
| #include "v9fs.h" |
| #include "cache.h" |
| |
| #define CACHETAG_LEN 11 |
| |
| struct fscache_netfs v9fs_cache_netfs = { |
| .name = "9p", |
| .version = 0, |
| }; |
| |
| /* |
| * v9fs_random_cachetag - Generate a random tag to be associated |
| * with a new cache session. |
| * |
| * The value of jiffies is used for a fairly randomly cache tag. |
| */ |
| |
| static |
| int v9fs_random_cachetag(struct v9fs_session_info *v9ses) |
| { |
| v9ses->cachetag = kmalloc(CACHETAG_LEN, GFP_KERNEL); |
| if (!v9ses->cachetag) |
| return -ENOMEM; |
| |
| return scnprintf(v9ses->cachetag, CACHETAG_LEN, "%lu", jiffies); |
| } |
| |
| const struct fscache_cookie_def v9fs_cache_session_index_def = { |
| .name = "9P.session", |
| .type = FSCACHE_COOKIE_TYPE_INDEX, |
| }; |
| |
| void v9fs_cache_session_get_cookie(struct v9fs_session_info *v9ses) |
| { |
| /* If no cache session tag was specified, we generate a random one. */ |
| if (!v9ses->cachetag) { |
| if (v9fs_random_cachetag(v9ses) < 0) { |
| v9ses->fscache = NULL; |
| kfree(v9ses->cachetag); |
| v9ses->cachetag = NULL; |
| return; |
| } |
| } |
| |
| v9ses->fscache = fscache_acquire_cookie(v9fs_cache_netfs.primary_index, |
| &v9fs_cache_session_index_def, |
| v9ses->cachetag, |
| strlen(v9ses->cachetag), |
| NULL, 0, |
| v9ses, 0, true); |
| p9_debug(P9_DEBUG_FSC, "session %p get cookie %p\n", |
| v9ses, v9ses->fscache); |
| } |
| |
| void v9fs_cache_session_put_cookie(struct v9fs_session_info *v9ses) |
| { |
| p9_debug(P9_DEBUG_FSC, "session %p put cookie %p\n", |
| v9ses, v9ses->fscache); |
| fscache_relinquish_cookie(v9ses->fscache, NULL, false); |
| v9ses->fscache = NULL; |
| } |
| |
| static enum |
| fscache_checkaux v9fs_cache_inode_check_aux(void *cookie_netfs_data, |
| const void *buffer, |
| uint16_t buflen, |
| loff_t object_size) |
| { |
| const struct v9fs_inode *v9inode = cookie_netfs_data; |
| |
| if (buflen != sizeof(v9inode->qid.version)) |
| return FSCACHE_CHECKAUX_OBSOLETE; |
| |
| if (memcmp(buffer, &v9inode->qid.version, |
| sizeof(v9inode->qid.version))) |
| return FSCACHE_CHECKAUX_OBSOLETE; |
| |
| return FSCACHE_CHECKAUX_OKAY; |
| } |
| |
| const struct fscache_cookie_def v9fs_cache_inode_index_def = { |
| .name = "9p.inode", |
| .type = FSCACHE_COOKIE_TYPE_DATAFILE, |
| .check_aux = v9fs_cache_inode_check_aux, |
| }; |
| |
| void v9fs_cache_inode_get_cookie(struct inode *inode) |
| { |
| struct v9fs_inode *v9inode; |
| struct v9fs_session_info *v9ses; |
| |
| if (!S_ISREG(inode->i_mode)) |
| return; |
| |
| v9inode = V9FS_I(inode); |
| if (v9inode->fscache) |
| return; |
| |
| v9ses = v9fs_inode2v9ses(inode); |
| v9inode->fscache = fscache_acquire_cookie(v9ses->fscache, |
| &v9fs_cache_inode_index_def, |
| &v9inode->qid.path, |
| sizeof(v9inode->qid.path), |
| &v9inode->qid.version, |
| sizeof(v9inode->qid.version), |
| v9inode, |
| i_size_read(&v9inode->vfs_inode), |
| true); |
| |
| p9_debug(P9_DEBUG_FSC, "inode %p get cookie %p\n", |
| inode, v9inode->fscache); |
| } |
| |
| void v9fs_cache_inode_put_cookie(struct inode *inode) |
| { |
| struct v9fs_inode *v9inode = V9FS_I(inode); |
| |
| if (!v9inode->fscache) |
| return; |
| p9_debug(P9_DEBUG_FSC, "inode %p put cookie %p\n", |
| inode, v9inode->fscache); |
| |
| fscache_relinquish_cookie(v9inode->fscache, &v9inode->qid.version, |
| false); |
| v9inode->fscache = NULL; |
| } |
| |
| void v9fs_cache_inode_flush_cookie(struct inode *inode) |
| { |
| struct v9fs_inode *v9inode = V9FS_I(inode); |
| |
| if (!v9inode->fscache) |
| return; |
| p9_debug(P9_DEBUG_FSC, "inode %p flush cookie %p\n", |
| inode, v9inode->fscache); |
| |
| fscache_relinquish_cookie(v9inode->fscache, NULL, true); |
| v9inode->fscache = NULL; |
| } |
| |
| void v9fs_cache_inode_set_cookie(struct inode *inode, struct file *filp) |
| { |
| struct v9fs_inode *v9inode = V9FS_I(inode); |
| |
| if (!v9inode->fscache) |
| return; |
| |
| mutex_lock(&v9inode->fscache_lock); |
| |
| if ((filp->f_flags & O_ACCMODE) != O_RDONLY) |
| v9fs_cache_inode_flush_cookie(inode); |
| else |
| v9fs_cache_inode_get_cookie(inode); |
| |
| mutex_unlock(&v9inode->fscache_lock); |
| } |
| |
| void v9fs_cache_inode_reset_cookie(struct inode *inode) |
| { |
| struct v9fs_inode *v9inode = V9FS_I(inode); |
| struct v9fs_session_info *v9ses; |
| struct fscache_cookie *old; |
| |
| if (!v9inode->fscache) |
| return; |
| |
| old = v9inode->fscache; |
| |
| mutex_lock(&v9inode->fscache_lock); |
| fscache_relinquish_cookie(v9inode->fscache, NULL, true); |
| |
| v9ses = v9fs_inode2v9ses(inode); |
| v9inode->fscache = fscache_acquire_cookie(v9ses->fscache, |
| &v9fs_cache_inode_index_def, |
| &v9inode->qid.path, |
| sizeof(v9inode->qid.path), |
| &v9inode->qid.version, |
| sizeof(v9inode->qid.version), |
| v9inode, |
| i_size_read(&v9inode->vfs_inode), |
| true); |
| p9_debug(P9_DEBUG_FSC, "inode %p revalidating cookie old %p new %p\n", |
| inode, old, v9inode->fscache); |
| |
| mutex_unlock(&v9inode->fscache_lock); |
| } |