Commit f89ce84b authored by Linus Torvalds's avatar Linus Torvalds

Merge tag '9p-for-5.16-rc1' of git://github.com/martinetd/linux

Pull 9p updates from Dominique Martinet:
 "Fixes, netfs read support and checkpatch rewrite:

   - fix syzcaller uninitialized value usage after missing error check

   - add module autoloading based on transport name

   - convert cached reads to use netfs helpers

   - adjust readahead based on transport msize

   - and many, many checkpatch.pl warning fixes..."

* tag '9p-for-5.16-rc1' of git://github.com/martinetd/linux:
  9p: fix a bunch of checkpatch warnings
  9p: set readahead and io size according to maxsize
  9p p9mode2perm: remove useless strlcpy and check sscanf return code
  9p v9fs_parse_options: replace simple_strtoul with kstrtouint
  9p: fix file headers
  fs/9p: fix indentation and Add missing a blank line after declaration
  fs/9p: fix warnings found by checkpatch.pl
  9p: fix minor indentation and codestyle
  fs/9p: cleanup: opening brace at the beginning of the next line
  9p: Convert to using the netfs helper lib to do reads and caching
  fscache_cookie_enabled: check cookie is valid before accessing it
  net/9p: autoload transport modules
  9p/net: fix missing error check in p9_check_errors
parents 59a2ceee 6e195b0f
...@@ -2,6 +2,7 @@ ...@@ -2,6 +2,7 @@
config 9P_FS config 9P_FS
tristate "Plan 9 Resource Sharing Support (9P2000)" tristate "Plan 9 Resource Sharing Support (9P2000)"
depends on INET && NET_9P depends on INET && NET_9P
select NETFS_SUPPORT
help help
If you say Y here, you will get experimental support for If you say Y here, you will get experimental support for
Plan 9 resource sharing via the 9P2000 protocol. Plan 9 resource sharing via the 9P2000 protocol.
......
// SPDX-License-Identifier: LGPL-2.1
/* /*
* Copyright IBM Corporation, 2010 * Copyright IBM Corporation, 2010
* Author Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com> * Author Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of version 2.1 of the GNU Lesser General Public License
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it would be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
*
*/ */
#include <linux/module.h> #include <linux/module.h>
...@@ -123,6 +115,7 @@ static int v9fs_set_acl(struct p9_fid *fid, int type, struct posix_acl *acl) ...@@ -123,6 +115,7 @@ static int v9fs_set_acl(struct p9_fid *fid, int type, struct posix_acl *acl)
char *name; char *name;
size_t size; size_t size;
void *buffer; void *buffer;
if (!acl) if (!acl)
return 0; return 0;
......
/* SPDX-License-Identifier: LGPL-2.1 */
/* /*
* Copyright IBM Corporation, 2010 * Copyright IBM Corporation, 2010
* Author Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com> * Author Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of version 2.1 of the GNU Lesser General Public License
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it would be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
*
*/ */
#ifndef FS_9P_ACL_H #ifndef FS_9P_ACL_H
#define FS_9P_ACL_H #define FS_9P_ACL_H
#ifdef CONFIG_9P_FS_POSIX_ACL #ifdef CONFIG_9P_FS_POSIX_ACL
extern int v9fs_get_acl(struct inode *, struct p9_fid *); int v9fs_get_acl(struct inode *inode, struct p9_fid *fid);
extern struct posix_acl *v9fs_iop_get_acl(struct inode *inode, int type, bool rcu); struct posix_acl *v9fs_iop_get_acl(struct inode *inode, int type,
extern int v9fs_acl_chmod(struct inode *, struct p9_fid *); bool rcu);
extern int v9fs_set_create_acl(struct inode *, struct p9_fid *, int v9fs_acl_chmod(struct inode *inode, struct p9_fid *fid);
struct posix_acl *, struct posix_acl *); int v9fs_set_create_acl(struct inode *inode, struct p9_fid *fid,
extern int v9fs_acl_mode(struct inode *dir, umode_t *modep, struct posix_acl *dacl, struct posix_acl *acl);
struct posix_acl **dpacl, struct posix_acl **pacl); int v9fs_acl_mode(struct inode *dir, umode_t *modep,
extern void v9fs_put_acl(struct posix_acl *dacl, struct posix_acl *acl); struct posix_acl **dpacl, struct posix_acl **pacl);
void v9fs_put_acl(struct posix_acl *dacl, struct posix_acl *acl);
#else #else
#define v9fs_iop_get_acl NULL #define v9fs_iop_get_acl NULL
static inline int v9fs_get_acl(struct inode *inode, struct p9_fid *fid) static inline int v9fs_get_acl(struct inode *inode, struct p9_fid *fid)
......
...@@ -19,8 +19,8 @@ ...@@ -19,8 +19,8 @@
#define CACHETAG_LEN 11 #define CACHETAG_LEN 11
struct fscache_netfs v9fs_cache_netfs = { struct fscache_netfs v9fs_cache_netfs = {
.name = "9p", .name = "9p",
.version = 0, .version = 0,
}; };
/* /*
...@@ -199,140 +199,3 @@ void v9fs_cache_inode_reset_cookie(struct inode *inode) ...@@ -199,140 +199,3 @@ void v9fs_cache_inode_reset_cookie(struct inode *inode)
mutex_unlock(&v9inode->fscache_lock); mutex_unlock(&v9inode->fscache_lock);
} }
int __v9fs_fscache_release_page(struct page *page, gfp_t gfp)
{
struct inode *inode = page->mapping->host;
struct v9fs_inode *v9inode = V9FS_I(inode);
BUG_ON(!v9inode->fscache);
return fscache_maybe_release_page(v9inode->fscache, page, gfp);
}
void __v9fs_fscache_invalidate_page(struct page *page)
{
struct inode *inode = page->mapping->host;
struct v9fs_inode *v9inode = V9FS_I(inode);
BUG_ON(!v9inode->fscache);
if (PageFsCache(page)) {
fscache_wait_on_page_write(v9inode->fscache, page);
BUG_ON(!PageLocked(page));
fscache_uncache_page(v9inode->fscache, page);
}
}
static void v9fs_vfs_readpage_complete(struct page *page, void *data,
int error)
{
if (!error)
SetPageUptodate(page);
unlock_page(page);
}
/*
* __v9fs_readpage_from_fscache - read a page from cache
*
* Returns 0 if the pages are in cache and a BIO is submitted,
* 1 if the pages are not in cache and -error otherwise.
*/
int __v9fs_readpage_from_fscache(struct inode *inode, struct page *page)
{
int ret;
const struct v9fs_inode *v9inode = V9FS_I(inode);
p9_debug(P9_DEBUG_FSC, "inode %p page %p\n", inode, page);
if (!v9inode->fscache)
return -ENOBUFS;
ret = fscache_read_or_alloc_page(v9inode->fscache,
page,
v9fs_vfs_readpage_complete,
NULL,
GFP_KERNEL);
switch (ret) {
case -ENOBUFS:
case -ENODATA:
p9_debug(P9_DEBUG_FSC, "page/inode not in cache %d\n", ret);
return 1;
case 0:
p9_debug(P9_DEBUG_FSC, "BIO submitted\n");
return ret;
default:
p9_debug(P9_DEBUG_FSC, "ret %d\n", ret);
return ret;
}
}
/*
* __v9fs_readpages_from_fscache - read multiple pages from cache
*
* Returns 0 if the pages are in cache and a BIO is submitted,
* 1 if the pages are not in cache and -error otherwise.
*/
int __v9fs_readpages_from_fscache(struct inode *inode,
struct address_space *mapping,
struct list_head *pages,
unsigned *nr_pages)
{
int ret;
const struct v9fs_inode *v9inode = V9FS_I(inode);
p9_debug(P9_DEBUG_FSC, "inode %p pages %u\n", inode, *nr_pages);
if (!v9inode->fscache)
return -ENOBUFS;
ret = fscache_read_or_alloc_pages(v9inode->fscache,
mapping, pages, nr_pages,
v9fs_vfs_readpage_complete,
NULL,
mapping_gfp_mask(mapping));
switch (ret) {
case -ENOBUFS:
case -ENODATA:
p9_debug(P9_DEBUG_FSC, "pages/inodes not in cache %d\n", ret);
return 1;
case 0:
BUG_ON(!list_empty(pages));
BUG_ON(*nr_pages != 0);
p9_debug(P9_DEBUG_FSC, "BIO submitted\n");
return ret;
default:
p9_debug(P9_DEBUG_FSC, "ret %d\n", ret);
return ret;
}
}
/*
* __v9fs_readpage_to_fscache - write a page to the cache
*
*/
void __v9fs_readpage_to_fscache(struct inode *inode, struct page *page)
{
int ret;
const struct v9fs_inode *v9inode = V9FS_I(inode);
p9_debug(P9_DEBUG_FSC, "inode %p page %p\n", inode, page);
ret = fscache_write_page(v9inode->fscache, page,
i_size_read(&v9inode->vfs_inode), GFP_KERNEL);
p9_debug(P9_DEBUG_FSC, "ret = %d\n", ret);
if (ret != 0)
v9fs_uncache_page(inode, page);
}
/*
* wait for a page to complete writing to the cache
*/
void __v9fs_fscache_wait_on_page_write(struct inode *inode, struct page *page)
{
const struct v9fs_inode *v9inode = V9FS_I(inode);
p9_debug(P9_DEBUG_FSC, "inode %p page %p\n", inode, page);
if (PageFsCache(page))
fscache_wait_on_page_write(v9inode->fscache, page);
}
...@@ -7,9 +7,10 @@ ...@@ -7,9 +7,10 @@
#ifndef _9P_CACHE_H #ifndef _9P_CACHE_H
#define _9P_CACHE_H #define _9P_CACHE_H
#ifdef CONFIG_9P_FSCACHE #define FSCACHE_USE_NEW_IO_API
#include <linux/fscache.h> #include <linux/fscache.h>
#include <linux/spinlock.h>
#ifdef CONFIG_9P_FSCACHE
extern struct fscache_netfs v9fs_cache_netfs; extern struct fscache_netfs v9fs_cache_netfs;
extern const struct fscache_cookie_def v9fs_cache_session_index_def; extern const struct fscache_cookie_def v9fs_cache_session_index_def;
...@@ -27,64 +28,6 @@ extern void v9fs_cache_inode_reset_cookie(struct inode *inode); ...@@ -27,64 +28,6 @@ extern void v9fs_cache_inode_reset_cookie(struct inode *inode);
extern int __v9fs_cache_register(void); extern int __v9fs_cache_register(void);
extern void __v9fs_cache_unregister(void); extern void __v9fs_cache_unregister(void);
extern int __v9fs_fscache_release_page(struct page *page, gfp_t gfp);
extern void __v9fs_fscache_invalidate_page(struct page *page);
extern int __v9fs_readpage_from_fscache(struct inode *inode,
struct page *page);
extern int __v9fs_readpages_from_fscache(struct inode *inode,
struct address_space *mapping,
struct list_head *pages,
unsigned *nr_pages);
extern void __v9fs_readpage_to_fscache(struct inode *inode, struct page *page);
extern void __v9fs_fscache_wait_on_page_write(struct inode *inode,
struct page *page);
static inline int v9fs_fscache_release_page(struct page *page,
gfp_t gfp)
{
return __v9fs_fscache_release_page(page, gfp);
}
static inline void v9fs_fscache_invalidate_page(struct page *page)
{
__v9fs_fscache_invalidate_page(page);
}
static inline int v9fs_readpage_from_fscache(struct inode *inode,
struct page *page)
{
return __v9fs_readpage_from_fscache(inode, page);
}
static inline int v9fs_readpages_from_fscache(struct inode *inode,
struct address_space *mapping,
struct list_head *pages,
unsigned *nr_pages)
{
return __v9fs_readpages_from_fscache(inode, mapping, pages,
nr_pages);
}
static inline void v9fs_readpage_to_fscache(struct inode *inode,
struct page *page)
{
if (PageFsCache(page))
__v9fs_readpage_to_fscache(inode, page);
}
static inline void v9fs_uncache_page(struct inode *inode, struct page *page)
{
struct v9fs_inode *v9inode = V9FS_I(inode);
fscache_uncache_page(v9inode->fscache, page);
BUG_ON(PageFsCache(page));
}
static inline void v9fs_fscache_wait_on_page_write(struct inode *inode,
struct page *page)
{
return __v9fs_fscache_wait_on_page_write(inode, page);
}
#else /* CONFIG_9P_FSCACHE */ #else /* CONFIG_9P_FSCACHE */
static inline void v9fs_cache_inode_get_cookie(struct inode *inode) static inline void v9fs_cache_inode_get_cookie(struct inode *inode)
...@@ -99,39 +42,5 @@ static inline void v9fs_cache_inode_set_cookie(struct inode *inode, struct file ...@@ -99,39 +42,5 @@ static inline void v9fs_cache_inode_set_cookie(struct inode *inode, struct file
{ {
} }
static inline int v9fs_fscache_release_page(struct page *page,
gfp_t gfp) {
return 1;
}
static inline void v9fs_fscache_invalidate_page(struct page *page) {}
static inline int v9fs_readpage_from_fscache(struct inode *inode,
struct page *page)
{
return -ENOBUFS;
}
static inline int v9fs_readpages_from_fscache(struct inode *inode,
struct address_space *mapping,
struct list_head *pages,
unsigned *nr_pages)
{
return -ENOBUFS;
}
static inline void v9fs_readpage_to_fscache(struct inode *inode,
struct page *page)
{}
static inline void v9fs_uncache_page(struct inode *inode, struct page *page)
{}
static inline void v9fs_fscache_wait_on_page_write(struct inode *inode,
struct page *page)
{
return;
}
#endif /* CONFIG_9P_FSCACHE */ #endif /* CONFIG_9P_FSCACHE */
#endif /* _9P_CACHE_H */ #endif /* _9P_CACHE_H */
...@@ -103,6 +103,7 @@ static struct p9_fid *v9fs_fid_find(struct dentry *dentry, kuid_t uid, int any) ...@@ -103,6 +103,7 @@ static struct p9_fid *v9fs_fid_find(struct dentry *dentry, kuid_t uid, int any)
/* we'll recheck under lock if there's anything to look in */ /* we'll recheck under lock if there's anything to look in */
if (!ret && dentry->d_fsdata) { if (!ret && dentry->d_fsdata) {
struct hlist_head *h = (struct hlist_head *)&dentry->d_fsdata; struct hlist_head *h = (struct hlist_head *)&dentry->d_fsdata;
spin_lock(&dentry->d_lock); spin_lock(&dentry->d_lock);
hlist_for_each_entry(fid, h, dlist) { hlist_for_each_entry(fid, h, dlist) {
if (any || uid_eq(fid->uid, uid)) { if (any || uid_eq(fid->uid, uid)) {
...@@ -185,7 +186,7 @@ static struct p9_fid *v9fs_fid_lookup_with_uid(struct dentry *dentry, ...@@ -185,7 +186,7 @@ static struct p9_fid *v9fs_fid_lookup_with_uid(struct dentry *dentry,
return ERR_PTR(-EPERM); return ERR_PTR(-EPERM);
if (v9fs_proto_dotu(v9ses) || v9fs_proto_dotl(v9ses)) if (v9fs_proto_dotu(v9ses) || v9fs_proto_dotl(v9ses))
uname = NULL; uname = NULL;
else else
uname = v9ses->uname; uname = v9ses->uname;
......
// SPDX-License-Identifier: GPL-2.0-only // SPDX-License-Identifier: GPL-2.0-only
/* /*
* linux/fs/9p/v9fs.c
*
* This file contains functions assisting in mapping VFS to 9P2000 * This file contains functions assisting in mapping VFS to 9P2000
* *
* Copyright (C) 2004-2008 by Eric Van Hensbergen <ericvh@gmail.com> * Copyright (C) 2004-2008 by Eric Van Hensbergen <ericvh@gmail.com>
...@@ -166,7 +164,7 @@ static int v9fs_parse_options(struct v9fs_session_info *v9ses, char *opts) ...@@ -166,7 +164,7 @@ static int v9fs_parse_options(struct v9fs_session_info *v9ses, char *opts)
substring_t args[MAX_OPT_ARGS]; substring_t args[MAX_OPT_ARGS];
char *p; char *p;
int option = 0; int option = 0;
char *s, *e; char *s;
int ret = 0; int ret = 0;
/* setup defaults */ /* setup defaults */
...@@ -190,8 +188,10 @@ static int v9fs_parse_options(struct v9fs_session_info *v9ses, char *opts) ...@@ -190,8 +188,10 @@ static int v9fs_parse_options(struct v9fs_session_info *v9ses, char *opts)
while ((p = strsep(&options, ",")) != NULL) { while ((p = strsep(&options, ",")) != NULL) {
int token, r; int token, r;
if (!*p) if (!*p)
continue; continue;
token = match_token(p, tokens, args); token = match_token(p, tokens, args);
switch (token) { switch (token) {
case Opt_debug: case Opt_debug:
...@@ -321,12 +321,13 @@ static int v9fs_parse_options(struct v9fs_session_info *v9ses, char *opts) ...@@ -321,12 +321,13 @@ static int v9fs_parse_options(struct v9fs_session_info *v9ses, char *opts)
v9ses->flags |= V9FS_ACCESS_CLIENT; v9ses->flags |= V9FS_ACCESS_CLIENT;
} else { } else {
uid_t uid; uid_t uid;
v9ses->flags |= V9FS_ACCESS_SINGLE; v9ses->flags |= V9FS_ACCESS_SINGLE;
uid = simple_strtoul(s, &e, 10); r = kstrtouint(s, 10, &uid);
if (*e != '\0') { if (r) {
ret = -EINVAL; ret = r;
pr_info("Unknown access argument %s\n", pr_info("Unknown access argument %s: %d\n",
s); s, r);
kfree(s); kfree(s);
continue; continue;
} }
...@@ -520,7 +521,8 @@ void v9fs_session_close(struct v9fs_session_info *v9ses) ...@@ -520,7 +521,8 @@ void v9fs_session_close(struct v9fs_session_info *v9ses)
* mark transport as disconnected and cancel all pending requests. * mark transport as disconnected and cancel all pending requests.
*/ */
void v9fs_session_cancel(struct v9fs_session_info *v9ses) { void v9fs_session_cancel(struct v9fs_session_info *v9ses)
{
p9_debug(P9_DEBUG_ERROR, "cancel session %p\n", v9ses); p9_debug(P9_DEBUG_ERROR, "cancel session %p\n", v9ses);
p9_client_disconnect(v9ses->clnt); p9_client_disconnect(v9ses->clnt);
} }
...@@ -659,6 +661,7 @@ static void v9fs_destroy_inode_cache(void) ...@@ -659,6 +661,7 @@ static void v9fs_destroy_inode_cache(void)
static int v9fs_cache_register(void) static int v9fs_cache_register(void)
{ {
int ret; int ret;
ret = v9fs_init_inode_cache(); ret = v9fs_init_inode_cache();
if (ret < 0) if (ret < 0)
return ret; return ret;
...@@ -686,6 +689,7 @@ static void v9fs_cache_unregister(void) ...@@ -686,6 +689,7 @@ static void v9fs_cache_unregister(void)
static int __init init_v9fs(void) static int __init init_v9fs(void)
{ {
int err; int err;
pr_info("Installing v9fs 9p2000 file system support\n"); pr_info("Installing v9fs 9p2000 file system support\n");
/* TODO: Setup list of registered trasnport modules */ /* TODO: Setup list of registered trasnport modules */
......
...@@ -124,15 +124,24 @@ static inline struct v9fs_inode *V9FS_I(const struct inode *inode) ...@@ -124,15 +124,24 @@ static inline struct v9fs_inode *V9FS_I(const struct inode *inode)
return container_of(inode, struct v9fs_inode, vfs_inode); return container_of(inode, struct v9fs_inode, vfs_inode);
} }
static inline struct fscache_cookie *v9fs_inode_cookie(struct v9fs_inode *v9inode)
{
#ifdef CONFIG_9P_FSCACHE
return v9inode->fscache;
#else
return NULL;
#endif
}
extern int v9fs_show_options(struct seq_file *m, struct dentry *root); extern int v9fs_show_options(struct seq_file *m, struct dentry *root);
struct p9_fid *v9fs_session_init(struct v9fs_session_info *, const char *, struct p9_fid *v9fs_session_init(struct v9fs_session_info *v9ses,
char *); const char *dev_name, char *data);
extern void v9fs_session_close(struct v9fs_session_info *v9ses); extern void v9fs_session_close(struct v9fs_session_info *v9ses);
extern void v9fs_session_cancel(struct v9fs_session_info *v9ses); extern void v9fs_session_cancel(struct v9fs_session_info *v9ses);
extern void v9fs_session_begin_cancel(struct v9fs_session_info *v9ses); extern void v9fs_session_begin_cancel(struct v9fs_session_info *v9ses);
extern struct dentry *v9fs_vfs_lookup(struct inode *dir, struct dentry *dentry, extern struct dentry *v9fs_vfs_lookup(struct inode *dir, struct dentry *dentry,
unsigned int flags); unsigned int flags);
extern int v9fs_vfs_unlink(struct inode *i, struct dentry *d); extern int v9fs_vfs_unlink(struct inode *i, struct dentry *d);
extern int v9fs_vfs_rmdir(struct inode *i, struct dentry *d); extern int v9fs_vfs_rmdir(struct inode *i, struct dentry *d);
extern int v9fs_vfs_rename(struct user_namespace *mnt_userns, extern int v9fs_vfs_rename(struct user_namespace *mnt_userns,
...@@ -158,7 +167,7 @@ extern struct inode *v9fs_inode_from_fid_dotl(struct v9fs_session_info *v9ses, ...@@ -158,7 +167,7 @@ extern struct inode *v9fs_inode_from_fid_dotl(struct v9fs_session_info *v9ses,
static inline struct v9fs_session_info *v9fs_inode2v9ses(struct inode *inode) static inline struct v9fs_session_info *v9fs_inode2v9ses(struct inode *inode)
{ {
return (inode->i_sb->s_fs_info); return inode->i_sb->s_fs_info;
} }
static inline struct v9fs_session_info *v9fs_dentry2v9ses(struct dentry *dentry) static inline struct v9fs_session_info *v9fs_dentry2v9ses(struct dentry *dentry)
......
...@@ -44,9 +44,10 @@ extern struct kmem_cache *v9fs_inode_cache; ...@@ -44,9 +44,10 @@ extern struct kmem_cache *v9fs_inode_cache;
struct inode *v9fs_alloc_inode(struct super_block *sb); struct inode *v9fs_alloc_inode(struct super_block *sb);
void v9fs_free_inode(struct inode *inode); void v9fs_free_inode(struct inode *inode);
struct inode *v9fs_get_inode(struct super_block *sb, umode_t mode, dev_t); struct inode *v9fs_get_inode(struct super_block *sb, umode_t mode,
dev_t rdev);
int v9fs_init_inode(struct v9fs_session_info *v9ses, int v9fs_init_inode(struct v9fs_session_info *v9ses,
struct inode *inode, umode_t mode, dev_t); struct inode *inode, umode_t mode, dev_t rdev);
void v9fs_evict_inode(struct inode *inode); void v9fs_evict_inode(struct inode *inode);
ino_t v9fs_qid2ino(struct p9_qid *qid); ino_t v9fs_qid2ino(struct p9_qid *qid);
void v9fs_stat2inode(struct p9_wstat *stat, struct inode *inode, void v9fs_stat2inode(struct p9_wstat *stat, struct inode *inode,
...@@ -59,8 +60,8 @@ void v9fs_inode2stat(struct inode *inode, struct p9_wstat *stat); ...@@ -59,8 +60,8 @@ void v9fs_inode2stat(struct inode *inode, struct p9_wstat *stat);
int v9fs_uflags2omode(int uflags, int extended); int v9fs_uflags2omode(int uflags, int extended);
void v9fs_blank_wstat(struct p9_wstat *wstat); void v9fs_blank_wstat(struct p9_wstat *wstat);
int v9fs_vfs_setattr_dotl(struct user_namespace *, struct dentry *, int v9fs_vfs_setattr_dotl(struct user_namespace *mnt_userns,
struct iattr *); struct dentry *dentry, struct iattr *iattr);
int v9fs_file_fsync_dotl(struct file *filp, loff_t start, loff_t end, int v9fs_file_fsync_dotl(struct file *filp, loff_t start, loff_t end,
int datasync); int datasync);
int v9fs_refresh_inode(struct p9_fid *fid, struct inode *inode); int v9fs_refresh_inode(struct p9_fid *fid, struct inode *inode);
...@@ -68,9 +69,9 @@ int v9fs_refresh_inode_dotl(struct p9_fid *fid, struct inode *inode); ...@@ -68,9 +69,9 @@ int v9fs_refresh_inode_dotl(struct p9_fid *fid, struct inode *inode);
static inline void v9fs_invalidate_inode_attr(struct inode *inode) static inline void v9fs_invalidate_inode_attr(struct inode *inode)
{ {
struct v9fs_inode *v9inode; struct v9fs_inode *v9inode;
v9inode = V9FS_I(inode); v9inode = V9FS_I(inode);
v9inode->cache_validity |= V9FS_INO_INVALID_ATTR; v9inode->cache_validity |= V9FS_INO_INVALID_ATTR;
return;
} }
int v9fs_open_to_dotl_flags(int flags); int v9fs_open_to_dotl_flags(int flags);
......
// SPDX-License-Identifier: GPL-2.0-only // SPDX-License-Identifier: GPL-2.0-only
/* /*
* linux/fs/9p/vfs_addr.c
*
* This file contians vfs address (mmap) ops for 9P2000. * This file contians vfs address (mmap) ops for 9P2000.
* *
* Copyright (C) 2005 by Eric Van Hensbergen <ericvh@gmail.com> * Copyright (C) 2005 by Eric Van Hensbergen <ericvh@gmail.com>
...@@ -19,7 +17,7 @@ ...@@ -19,7 +17,7 @@
#include <linux/idr.h> #include <linux/idr.h>
#include <linux/sched.h> #include <linux/sched.h>
#include <linux/uio.h> #include <linux/uio.h>
#include <linux/bvec.h> #include <linux/netfs.h>
#include <net/9p/9p.h> #include <net/9p/9p.h>
#include <net/9p/client.h> #include <net/9p/client.h>
...@@ -29,88 +27,97 @@ ...@@ -29,88 +27,97 @@
#include "fid.h" #include "fid.h"
/** /**
* v9fs_fid_readpage - read an entire page in from 9P * v9fs_req_issue_op - Issue a read from 9P
* @data: Opaque pointer to the fid being read * @subreq: The read to make
* @page: structure to page
*
*/ */
static int v9fs_fid_readpage(void *data, struct page *page) static void v9fs_req_issue_op(struct netfs_read_subrequest *subreq)
{ {
struct p9_fid *fid = data; struct netfs_read_request *rreq = subreq->rreq;
struct inode *inode = page->mapping->host; struct p9_fid *fid = rreq->netfs_priv;
struct bio_vec bvec = {.bv_page = page, .bv_len = PAGE_SIZE};
struct iov_iter to; struct iov_iter to;
int retval, err; loff_t pos = subreq->start + subreq->transferred;
size_t len = subreq->len - subreq->transferred;
int total, err;
p9_debug(P9_DEBUG_VFS, "\n"); iov_iter_xarray(&to, READ, &rreq->mapping->i_pages, pos, len);
BUG_ON(!PageLocked(page)); total = p9_client_read(fid, pos, &to, &err);
netfs_subreq_terminated(subreq, err ?: total, false);
}
retval = v9fs_readpage_from_fscache(inode, page); /**
if (retval == 0) * v9fs_init_rreq - Initialise a read request
return retval; * @rreq: The read request
* @file: The file being read from
*/
static void v9fs_init_rreq(struct netfs_read_request *rreq, struct file *file)
{
struct p9_fid *fid = file->private_data;
iov_iter_bvec(&to, READ, &bvec, 1, PAGE_SIZE); refcount_inc(&fid->count);
rreq->netfs_priv = fid;
}
retval = p9_client_read(fid, page_offset(page), &to, &err); /**
if (err) { * v9fs_req_cleanup - Cleanup request initialized by v9fs_init_rreq
v9fs_uncache_page(inode, page); * @mapping: unused mapping of request to cleanup
retval = err; * @priv: private data to cleanup, a fid, guaranted non-null.
goto done; */
} static void v9fs_req_cleanup(struct address_space *mapping, void *priv)
{
struct p9_fid *fid = priv;
zero_user(page, retval, PAGE_SIZE - retval); p9_client_clunk(fid);
flush_dcache_page(page); }
SetPageUptodate(page);
v9fs_readpage_to_fscache(inode, page); /**
retval = 0; * v9fs_is_cache_enabled - Determine if caching is enabled for an inode
* @inode: The inode to check
*/
static bool v9fs_is_cache_enabled(struct inode *inode)
{
struct fscache_cookie *cookie = v9fs_inode_cookie(V9FS_I(inode));
done: return fscache_cookie_enabled(cookie) && !hlist_empty(&cookie->backing_objects);
unlock_page(page);
return retval;
} }
/**
* v9fs_begin_cache_operation - Begin a cache operation for a read
* @rreq: The read request
*/
static int v9fs_begin_cache_operation(struct netfs_read_request *rreq)
{
struct fscache_cookie *cookie = v9fs_inode_cookie(V9FS_I(rreq->inode));
return fscache_begin_read_operation(rreq, cookie);
}
static const struct netfs_read_request_ops v9fs_req_ops = {
.init_rreq = v9fs_init_rreq,
.is_cache_enabled = v9fs_is_cache_enabled,
.begin_cache_operation = v9fs_begin_cache_operation,
.issue_op = v9fs_req_issue_op,
.cleanup = v9fs_req_cleanup,
};
/** /**
* v9fs_vfs_readpage - read an entire page in from 9P * v9fs_vfs_readpage - read an entire page in from 9P
* * @file: file being read
* @filp: file being read
* @page: structure to page * @page: structure to page
* *
*/ */
static int v9fs_vfs_readpage(struct file *file, struct page *page)
static int v9fs_vfs_readpage(struct file *filp, struct page *page)
{ {
return v9fs_fid_readpage(filp->private_data, page); return netfs_readpage(file, page, &v9fs_req_ops, NULL);
} }
/** /**
* v9fs_vfs_readpages - read a set of pages from 9P * v9fs_vfs_readahead - read a set of pages from 9P
* * @ractl: The readahead parameters
* @filp: file being read
* @mapping: the address space
* @pages: list of pages to read
* @nr_pages: count of pages to read
*
*/ */
static void v9fs_vfs_readahead(struct readahead_control *ractl)
static int v9fs_vfs_readpages(struct file *filp, struct address_space *mapping,
struct list_head *pages, unsigned nr_pages)
{ {
int ret = 0; netfs_readahead(ractl, &v9fs_req_ops, NULL);
struct inode *inode;
inode = mapping->host;
p9_debug(P9_DEBUG_VFS, "inode: %p file: %p\n", inode, filp);
ret = v9fs_readpages_from_fscache(inode, mapping, pages, &nr_pages);
if (ret == 0)
return ret;
ret = read_cache_pages(mapping, pages, v9fs_fid_readpage,
filp->private_data);
p9_debug(P9_DEBUG_VFS, " = %d\n", ret);
return ret;
} }
/** /**
...@@ -125,7 +132,14 @@ static int v9fs_release_page(struct page *page, gfp_t gfp) ...@@ -125,7 +132,14 @@ static int v9fs_release_page(struct page *page, gfp_t gfp)
{ {
if (PagePrivate(page)) if (PagePrivate(page))
return 0; return 0;
return v9fs_fscache_release_page(page, gfp); #ifdef CONFIG_9P_FSCACHE
if (PageFsCache(page)) {
if (!(gfp & __GFP_DIRECT_RECLAIM) || !(gfp & __GFP_FS))
return 0;
wait_on_page_fscache(page);
}
#endif
return 1;
} }
/** /**
...@@ -138,21 +152,16 @@ static int v9fs_release_page(struct page *page, gfp_t gfp) ...@@ -138,21 +152,16 @@ static int v9fs_release_page(struct page *page, gfp_t gfp)
static void v9fs_invalidate_page(struct page *page, unsigned int offset, static void v9fs_invalidate_page(struct page *page, unsigned int offset,
unsigned int length) unsigned int length)
{ {
/* wait_on_page_fscache(page);
* If called with zero offset, we should release
* the private state assocated with the page
*/
if (offset == 0 && length == PAGE_SIZE)
v9fs_fscache_invalidate_page(page);
} }
static int v9fs_vfs_writepage_locked(struct page *page) static int v9fs_vfs_writepage_locked(struct page *page)
{ {
struct inode *inode = page->mapping->host; struct inode *inode = page->mapping->host;
struct v9fs_inode *v9inode = V9FS_I(inode); struct v9fs_inode *v9inode = V9FS_I(inode);
loff_t start = page_offset(page);
loff_t size = i_size_read(inode); loff_t size = i_size_read(inode);
struct iov_iter from; struct iov_iter from;
struct bio_vec bvec;
int err, len; int err, len;
if (page->index == size >> PAGE_SHIFT) if (page->index == size >> PAGE_SHIFT)
...@@ -160,17 +169,14 @@ static int v9fs_vfs_writepage_locked(struct page *page) ...@@ -160,17 +169,14 @@ static int v9fs_vfs_writepage_locked(struct page *page)
else else
len = PAGE_SIZE; len = PAGE_SIZE;
bvec.bv_page = page; iov_iter_xarray(&from, WRITE, &page->mapping->i_pages, start, len);
bvec.bv_offset = 0;
bvec.bv_len = len;
iov_iter_bvec(&from, WRITE, &bvec, 1, len);
/* We should have writeback_fid always set */ /* We should have writeback_fid always set */
BUG_ON(!v9inode->writeback_fid); BUG_ON(!v9inode->writeback_fid);
set_page_writeback(page); set_page_writeback(page);
p9_client_write(v9inode->writeback_fid, page_offset(page), &from, &err); p9_client_write(v9inode->writeback_fid, start, &from, &err);
end_page_writeback(page); end_page_writeback(page);
return err; return err;
...@@ -208,14 +214,13 @@ static int v9fs_vfs_writepage(struct page *page, struct writeback_control *wbc) ...@@ -208,14 +214,13 @@ static int v9fs_vfs_writepage(struct page *page, struct writeback_control *wbc)
static int v9fs_launder_page(struct page *page) static int v9fs_launder_page(struct page *page)
{ {
int retval; int retval;
struct inode *inode = page->mapping->host;
v9fs_fscache_wait_on_page_write(inode, page);
if (clear_page_dirty_for_io(page)) { if (clear_page_dirty_for_io(page)) {
retval = v9fs_vfs_writepage_locked(page); retval = v9fs_vfs_writepage_locked(page);
if (retval) if (retval)
return retval; return retval;
} }
wait_on_page_fscache(page);
return 0; return 0;
} }
...@@ -242,11 +247,13 @@ v9fs_direct_IO(struct kiocb *iocb, struct iov_iter *iter) ...@@ -242,11 +247,13 @@ v9fs_direct_IO(struct kiocb *iocb, struct iov_iter *iter)
loff_t pos = iocb->ki_pos; loff_t pos = iocb->ki_pos;
ssize_t n; ssize_t n;
int err = 0; int err = 0;
if (iov_iter_rw(iter) == WRITE) { if (iov_iter_rw(iter) == WRITE) {
n = p9_client_write(file->private_data, pos, iter, &err); n = p9_client_write(file->private_data, pos, iter, &err);
if (n) { if (n) {
struct inode *inode = file_inode(file); struct inode *inode = file_inode(file);
loff_t i_size = i_size_read(inode); loff_t i_size = i_size_read(inode);
if (pos + n > i_size) if (pos + n > i_size)
inode_add_bytes(inode, pos + n - i_size); inode_add_bytes(inode, pos + n - i_size);
} }
...@@ -257,43 +264,32 @@ v9fs_direct_IO(struct kiocb *iocb, struct iov_iter *iter) ...@@ -257,43 +264,32 @@ v9fs_direct_IO(struct kiocb *iocb, struct iov_iter *iter)
} }
static int v9fs_write_begin(struct file *filp, struct address_space *mapping, static int v9fs_write_begin(struct file *filp, struct address_space *mapping,
loff_t pos, unsigned len, unsigned flags, loff_t pos, unsigned int len, unsigned int flags,
struct page **pagep, void **fsdata) struct page **pagep, void **fsdata)
{ {
int retval = 0; int retval;
struct page *page; struct page *page;
struct v9fs_inode *v9inode; struct v9fs_inode *v9inode = V9FS_I(mapping->host);
pgoff_t index = pos >> PAGE_SHIFT;
struct inode *inode = mapping->host;
p9_debug(P9_DEBUG_VFS, "filp %p, mapping %p\n", filp, mapping); p9_debug(P9_DEBUG_VFS, "filp %p, mapping %p\n", filp, mapping);
v9inode = V9FS_I(inode);
start:
page = grab_cache_page_write_begin(mapping, index, flags);
if (!page) {
retval = -ENOMEM;
goto out;
}
BUG_ON(!v9inode->writeback_fid); BUG_ON(!v9inode->writeback_fid);
if (PageUptodate(page))
goto out;
if (len == PAGE_SIZE) /* Prefetch area to be written into the cache if we're caching this
goto out; * file. We need to do this before we get a lock on the page in case
* there's more than one writer competing for the same cache block.
*/
retval = netfs_write_begin(filp, mapping, pos, len, flags, &page, fsdata,
&v9fs_req_ops, NULL);
if (retval < 0)
return retval;
retval = v9fs_fid_readpage(v9inode->writeback_fid, page); *pagep = find_subpage(page, pos / PAGE_SIZE);
put_page(page);
if (!retval)
goto start;
out:
*pagep = page;
return retval; return retval;
} }
static int v9fs_write_end(struct file *filp, struct address_space *mapping, static int v9fs_write_end(struct file *filp, struct address_space *mapping,
loff_t pos, unsigned len, unsigned copied, loff_t pos, unsigned int len, unsigned int copied,
struct page *page, void *fsdata) struct page *page, void *fsdata)
{ {
loff_t last_pos = pos + copied; loff_t last_pos = pos + copied;
...@@ -305,10 +301,11 @@ static int v9fs_write_end(struct file *filp, struct address_space *mapping, ...@@ -305,10 +301,11 @@ static int v9fs_write_end(struct file *filp, struct address_space *mapping,
if (unlikely(copied < len)) { if (unlikely(copied < len)) {
copied = 0; copied = 0;
goto out; goto out;
} else if (len == PAGE_SIZE) {
SetPageUptodate(page);
} }
SetPageUptodate(page);
} }
/* /*
* No need to use i_size_read() here, the i_size * No need to use i_size_read() here, the i_size
* cannot change under us because we hold the i_mutex. * cannot change under us because we hold the i_mutex.
...@@ -328,7 +325,7 @@ static int v9fs_write_end(struct file *filp, struct address_space *mapping, ...@@ -328,7 +325,7 @@ static int v9fs_write_end(struct file *filp, struct address_space *mapping,
const struct address_space_operations v9fs_addr_operations = { const struct address_space_operations v9fs_addr_operations = {
.readpage = v9fs_vfs_readpage, .readpage = v9fs_vfs_readpage,
.readpages = v9fs_vfs_readpages, .readahead = v9fs_vfs_readahead,
.set_page_dirty = __set_page_dirty_nobuffers, .set_page_dirty = __set_page_dirty_nobuffers,
.writepage = v9fs_vfs_writepage, .writepage = v9fs_vfs_writepage,
.write_begin = v9fs_write_begin, .write_begin = v9fs_write_begin,
......
// SPDX-License-Identifier: GPL-2.0-only // SPDX-License-Identifier: GPL-2.0-only
/* /*
* linux/fs/9p/vfs_dentry.c
*
* This file contians vfs dentry ops for the 9P2000 protocol. * This file contians vfs dentry ops for the 9P2000 protocol.
* *
* Copyright (C) 2004 by Eric Van Hensbergen <ericvh@gmail.com> * Copyright (C) 2004 by Eric Van Hensbergen <ericvh@gmail.com>
...@@ -52,6 +50,7 @@ static int v9fs_cached_dentry_delete(const struct dentry *dentry) ...@@ -52,6 +50,7 @@ static int v9fs_cached_dentry_delete(const struct dentry *dentry)
static void v9fs_dentry_release(struct dentry *dentry) static void v9fs_dentry_release(struct dentry *dentry)
{ {
struct hlist_node *p, *n; struct hlist_node *p, *n;
p9_debug(P9_DEBUG_VFS, " dentry: %pd (%p)\n", p9_debug(P9_DEBUG_VFS, " dentry: %pd (%p)\n",
dentry, dentry); dentry, dentry);
hlist_for_each_safe(p, n, (struct hlist_head *)&dentry->d_fsdata) hlist_for_each_safe(p, n, (struct hlist_head *)&dentry->d_fsdata)
...@@ -76,6 +75,7 @@ static int v9fs_lookup_revalidate(struct dentry *dentry, unsigned int flags) ...@@ -76,6 +75,7 @@ static int v9fs_lookup_revalidate(struct dentry *dentry, unsigned int flags)
if (v9inode->cache_validity & V9FS_INO_INVALID_ATTR) { if (v9inode->cache_validity & V9FS_INO_INVALID_ATTR) {
int retval; int retval;
struct v9fs_session_info *v9ses; struct v9fs_session_info *v9ses;
fid = v9fs_fid_lookup(dentry); fid = v9fs_fid_lookup(dentry);
if (IS_ERR(fid)) if (IS_ERR(fid))
return PTR_ERR(fid); return PTR_ERR(fid);
......
// SPDX-License-Identifier: GPL-2.0-only // SPDX-License-Identifier: GPL-2.0-only
/* /*
* linux/fs/9p/vfs_dir.c
*
* This file contains vfs directory ops for the 9P2000 protocol. * This file contains vfs directory ops for the 9P2000 protocol.
* *
* Copyright (C) 2004 by Eric Van Hensbergen <ericvh@gmail.com> * Copyright (C) 2004 by Eric Van Hensbergen <ericvh@gmail.com>
...@@ -71,6 +69,7 @@ static inline int dt_type(struct p9_wstat *mistat) ...@@ -71,6 +69,7 @@ static inline int dt_type(struct p9_wstat *mistat)
static struct p9_rdir *v9fs_alloc_rdir_buf(struct file *filp, int buflen) static struct p9_rdir *v9fs_alloc_rdir_buf(struct file *filp, int buflen)
{ {
struct p9_fid *fid = filp->private_data; struct p9_fid *fid = filp->private_data;
if (!fid->rdir) if (!fid->rdir)
fid->rdir = kzalloc(sizeof(struct p9_rdir) + buflen, GFP_KERNEL); fid->rdir = kzalloc(sizeof(struct p9_rdir) + buflen, GFP_KERNEL);
return fid->rdir; return fid->rdir;
...@@ -108,6 +107,7 @@ static int v9fs_dir_readdir(struct file *file, struct dir_context *ctx) ...@@ -108,6 +107,7 @@ static int v9fs_dir_readdir(struct file *file, struct dir_context *ctx)
if (rdir->tail == rdir->head) { if (rdir->tail == rdir->head) {
struct iov_iter to; struct iov_iter to;
int n; int n;
iov_iter_kvec(&to, READ, &kvec, 1, buflen); iov_iter_kvec(&to, READ, &kvec, 1, buflen);
n = p9_client_read(file->private_data, ctx->pos, &to, n = p9_client_read(file->private_data, ctx->pos, &to,
&err); &err);
...@@ -233,5 +233,5 @@ const struct file_operations v9fs_dir_operations_dotl = { ...@@ -233,5 +233,5 @@ const struct file_operations v9fs_dir_operations_dotl = {
.iterate_shared = v9fs_dir_readdir_dotl, .iterate_shared = v9fs_dir_readdir_dotl,
.open = v9fs_file_open, .open = v9fs_file_open,
.release = v9fs_dir_release, .release = v9fs_dir_release,
.fsync = v9fs_file_fsync_dotl, .fsync = v9fs_file_fsync_dotl,
}; };
// SPDX-License-Identifier: GPL-2.0-only // SPDX-License-Identifier: GPL-2.0-only
/* /*
* linux/fs/9p/vfs_file.c
*
* This file contians vfs file ops for 9P2000. * This file contians vfs file ops for 9P2000.
* *
* Copyright (C) 2004 by Eric Van Hensbergen <ericvh@gmail.com> * Copyright (C) 2004 by Eric Van Hensbergen <ericvh@gmail.com>
...@@ -408,6 +406,7 @@ v9fs_file_write_iter(struct kiocb *iocb, struct iov_iter *from) ...@@ -408,6 +406,7 @@ v9fs_file_write_iter(struct kiocb *iocb, struct iov_iter *from)
struct inode *inode = file_inode(file); struct inode *inode = file_inode(file);
loff_t i_size; loff_t i_size;
unsigned long pg_start, pg_end; unsigned long pg_start, pg_end;
pg_start = origin >> PAGE_SHIFT; pg_start = origin >> PAGE_SHIFT;
pg_end = (origin + retval - 1) >> PAGE_SHIFT; pg_end = (origin + retval - 1) >> PAGE_SHIFT;
if (inode->i_mapping && inode->i_mapping->nrpages) if (inode->i_mapping && inode->i_mapping->nrpages)
...@@ -537,14 +536,23 @@ v9fs_vm_page_mkwrite(struct vm_fault *vmf) ...@@ -537,14 +536,23 @@ v9fs_vm_page_mkwrite(struct vm_fault *vmf)
p9_debug(P9_DEBUG_VFS, "page %p fid %lx\n", p9_debug(P9_DEBUG_VFS, "page %p fid %lx\n",
page, (unsigned long)filp->private_data); page, (unsigned long)filp->private_data);
v9inode = V9FS_I(inode);
/* Wait for the page to be written to the cache before we allow it to
* be modified. We then assume the entire page will need writing back.
*/
#ifdef CONFIG_9P_FSCACHE
if (PageFsCache(page) &&
wait_on_page_fscache_killable(page) < 0)
return VM_FAULT_RETRY;
#endif
/* Update file times before taking page lock */ /* Update file times before taking page lock */
file_update_time(filp); file_update_time(filp);
v9inode = V9FS_I(inode);
/* make sure the cache has finished storing the page */
v9fs_fscache_wait_on_page_write(inode, page);
BUG_ON(!v9inode->writeback_fid); BUG_ON(!v9inode->writeback_fid);
lock_page(page); if (lock_page_killable(page) < 0)
return VM_FAULT_RETRY;
if (page->mapping != inode->i_mapping) if (page->mapping != inode->i_mapping)
goto out_unlock; goto out_unlock;
wait_for_stable_page(page); wait_for_stable_page(page);
......
// SPDX-License-Identifier: GPL-2.0-only // SPDX-License-Identifier: GPL-2.0-only
/* /*
* linux/fs/9p/vfs_inode.c
*
* This file contains vfs inode ops for the 9P2000 protocol. * This file contains vfs inode ops for the 9P2000 protocol.
* *
* Copyright (C) 2004 by Eric Van Hensbergen <ericvh@gmail.com> * Copyright (C) 2004 by Eric Van Hensbergen <ericvh@gmail.com>
...@@ -49,6 +47,7 @@ static const struct inode_operations v9fs_symlink_inode_operations; ...@@ -49,6 +47,7 @@ static const struct inode_operations v9fs_symlink_inode_operations;
static u32 unixmode2p9mode(struct v9fs_session_info *v9ses, umode_t mode) static u32 unixmode2p9mode(struct v9fs_session_info *v9ses, umode_t mode)
{ {
int res; int res;
res = mode & 0777; res = mode & 0777;
if (S_ISDIR(mode)) if (S_ISDIR(mode))
res |= P9_DMDIR; res |= P9_DMDIR;
...@@ -110,7 +109,7 @@ static int p9mode2perm(struct v9fs_session_info *v9ses, ...@@ -110,7 +109,7 @@ static int p9mode2perm(struct v9fs_session_info *v9ses,
static umode_t p9mode2unixmode(struct v9fs_session_info *v9ses, static umode_t p9mode2unixmode(struct v9fs_session_info *v9ses,
struct p9_wstat *stat, dev_t *rdev) struct p9_wstat *stat, dev_t *rdev)
{ {
int res; int res, r;
u32 mode = stat->mode; u32 mode = stat->mode;
*rdev = 0; *rdev = 0;
...@@ -128,11 +127,16 @@ static umode_t p9mode2unixmode(struct v9fs_session_info *v9ses, ...@@ -128,11 +127,16 @@ static umode_t p9mode2unixmode(struct v9fs_session_info *v9ses,
res |= S_IFIFO; res |= S_IFIFO;
else if ((mode & P9_DMDEVICE) && (v9fs_proto_dotu(v9ses)) else if ((mode & P9_DMDEVICE) && (v9fs_proto_dotu(v9ses))
&& (v9ses->nodev == 0)) { && (v9ses->nodev == 0)) {
char type = 0, ext[32]; char type = 0;
int major = -1, minor = -1; int major = -1, minor = -1;
strlcpy(ext, stat->extension, sizeof(ext)); r = sscanf(stat->extension, "%c %i %i", &type, &major, &minor);
sscanf(ext, "%c %i %i", &type, &major, &minor); if (r != 3) {
p9_debug(P9_DEBUG_ERROR,
"invalid device string, umode will be bogus: %s\n",
stat->extension);
return res;
}
switch (type) { switch (type) {
case 'c': case 'c':
res |= S_IFCHR; res |= S_IFCHR;
...@@ -223,6 +227,7 @@ v9fs_blank_wstat(struct p9_wstat *wstat) ...@@ -223,6 +227,7 @@ v9fs_blank_wstat(struct p9_wstat *wstat)
struct inode *v9fs_alloc_inode(struct super_block *sb) struct inode *v9fs_alloc_inode(struct super_block *sb)
{ {
struct v9fs_inode *v9inode; struct v9fs_inode *v9inode;
v9inode = kmem_cache_alloc(v9fs_inode_cache, GFP_KERNEL); v9inode = kmem_cache_alloc(v9fs_inode_cache, GFP_KERNEL);
if (!v9inode) if (!v9inode)
return NULL; return NULL;
...@@ -251,7 +256,7 @@ int v9fs_init_inode(struct v9fs_session_info *v9ses, ...@@ -251,7 +256,7 @@ int v9fs_init_inode(struct v9fs_session_info *v9ses,
{ {
int err = 0; int err = 0;
inode_init_owner(&init_user_ns,inode, NULL, mode); inode_init_owner(&init_user_ns, inode, NULL, mode);
inode->i_blocks = 0; inode->i_blocks = 0;
inode->i_rdev = rdev; inode->i_rdev = rdev;
inode->i_atime = inode->i_mtime = inode->i_ctime = current_time(inode); inode->i_atime = inode->i_mtime = inode->i_ctime = current_time(inode);
...@@ -440,7 +445,7 @@ static struct inode *v9fs_qid_iget(struct super_block *sb, ...@@ -440,7 +445,7 @@ static struct inode *v9fs_qid_iget(struct super_block *sb,
unsigned long i_ino; unsigned long i_ino;
struct inode *inode; struct inode *inode;
struct v9fs_session_info *v9ses = sb->s_fs_info; struct v9fs_session_info *v9ses = sb->s_fs_info;
int (*test)(struct inode *, void *); int (*test)(struct inode *inode, void *data);
if (new) if (new)
test = v9fs_test_new_inode; test = v9fs_test_new_inode;
...@@ -499,8 +504,10 @@ v9fs_inode_from_fid(struct v9fs_session_info *v9ses, struct p9_fid *fid, ...@@ -499,8 +504,10 @@ v9fs_inode_from_fid(struct v9fs_session_info *v9ses, struct p9_fid *fid,
static int v9fs_at_to_dotl_flags(int flags) static int v9fs_at_to_dotl_flags(int flags)
{ {
int rflags = 0; int rflags = 0;
if (flags & AT_REMOVEDIR) if (flags & AT_REMOVEDIR)
rflags |= P9_DOTL_AT_REMOVEDIR; rflags |= P9_DOTL_AT_REMOVEDIR;
return rflags; return rflags;
} }
...@@ -797,7 +804,7 @@ struct dentry *v9fs_vfs_lookup(struct inode *dir, struct dentry *dentry, ...@@ -797,7 +804,7 @@ struct dentry *v9fs_vfs_lookup(struct inode *dir, struct dentry *dentry,
static int static int
v9fs_vfs_atomic_open(struct inode *dir, struct dentry *dentry, v9fs_vfs_atomic_open(struct inode *dir, struct dentry *dentry,
struct file *file, unsigned flags, umode_t mode) struct file *file, unsigned int flags, umode_t mode)
{ {
int err; int err;
u32 perm; u32 perm;
...@@ -1084,7 +1091,7 @@ static int v9fs_vfs_setattr(struct user_namespace *mnt_userns, ...@@ -1084,7 +1091,7 @@ static int v9fs_vfs_setattr(struct user_namespace *mnt_userns,
fid = v9fs_fid_lookup(dentry); fid = v9fs_fid_lookup(dentry);
use_dentry = 1; use_dentry = 1;
} }
if(IS_ERR(fid)) if (IS_ERR(fid))
return PTR_ERR(fid); return PTR_ERR(fid);
v9fs_blank_wstat(&wstat); v9fs_blank_wstat(&wstat);
...@@ -1364,7 +1371,7 @@ v9fs_vfs_mknod(struct user_namespace *mnt_userns, struct inode *dir, ...@@ -1364,7 +1371,7 @@ v9fs_vfs_mknod(struct user_namespace *mnt_userns, struct inode *dir,
char name[2 + U32_MAX_DIGITS + 1 + U32_MAX_DIGITS + 1]; char name[2 + U32_MAX_DIGITS + 1 + U32_MAX_DIGITS + 1];
u32 perm; u32 perm;
p9_debug(P9_DEBUG_VFS, " %lu,%pd mode: %hx MAJOR: %u MINOR: %u\n", p9_debug(P9_DEBUG_VFS, " %lu,%pd mode: %x MAJOR: %u MINOR: %u\n",
dir->i_ino, dentry, mode, dir->i_ino, dentry, mode,
MAJOR(rdev), MINOR(rdev)); MAJOR(rdev), MINOR(rdev));
......
// SPDX-License-Identifier: GPL-2.0-only // SPDX-License-Identifier: GPL-2.0-only
/* /*
* linux/fs/9p/vfs_inode_dotl.c
*
* This file contains vfs inode ops for the 9P2000.L protocol. * This file contains vfs inode ops for the 9P2000.L protocol.
* *
* Copyright (C) 2004 by Eric Van Hensbergen <ericvh@gmail.com> * Copyright (C) 2004 by Eric Van Hensbergen <ericvh@gmail.com>
...@@ -107,7 +105,7 @@ static struct inode *v9fs_qid_iget_dotl(struct super_block *sb, ...@@ -107,7 +105,7 @@ static struct inode *v9fs_qid_iget_dotl(struct super_block *sb,
unsigned long i_ino; unsigned long i_ino;
struct inode *inode; struct inode *inode;
struct v9fs_session_info *v9ses = sb->s_fs_info; struct v9fs_session_info *v9ses = sb->s_fs_info;
int (*test)(struct inode *, void *); int (*test)(struct inode *inode, void *data);
if (new) if (new)
test = v9fs_test_new_inode_dotl; test = v9fs_test_new_inode_dotl;
...@@ -230,7 +228,7 @@ v9fs_vfs_create_dotl(struct user_namespace *mnt_userns, struct inode *dir, ...@@ -230,7 +228,7 @@ v9fs_vfs_create_dotl(struct user_namespace *mnt_userns, struct inode *dir,
static int static int
v9fs_vfs_atomic_open_dotl(struct inode *dir, struct dentry *dentry, v9fs_vfs_atomic_open_dotl(struct inode *dir, struct dentry *dentry,
struct file *file, unsigned flags, umode_t omode) struct file *file, unsigned int flags, umode_t omode)
{ {
int err = 0; int err = 0;
kgid_t gid; kgid_t gid;
...@@ -261,7 +259,7 @@ v9fs_vfs_atomic_open_dotl(struct inode *dir, struct dentry *dentry, ...@@ -261,7 +259,7 @@ v9fs_vfs_atomic_open_dotl(struct inode *dir, struct dentry *dentry,
v9ses = v9fs_inode2v9ses(dir); v9ses = v9fs_inode2v9ses(dir);
name = dentry->d_name.name; name = dentry->d_name.name;
p9_debug(P9_DEBUG_VFS, "name:%s flags:0x%x mode:0x%hx\n", p9_debug(P9_DEBUG_VFS, "name:%s flags:0x%x mode:0x%x\n",
name, flags, omode); name, flags, omode);
dfid = v9fs_parent_fid(dentry); dfid = v9fs_parent_fid(dentry);
...@@ -807,6 +805,7 @@ v9fs_vfs_link_dotl(struct dentry *old_dentry, struct inode *dir, ...@@ -807,6 +805,7 @@ v9fs_vfs_link_dotl(struct dentry *old_dentry, struct inode *dir,
if (v9ses->cache == CACHE_LOOSE || v9ses->cache == CACHE_FSCACHE) { if (v9ses->cache == CACHE_LOOSE || v9ses->cache == CACHE_FSCACHE) {
/* Get the latest stat info from server. */ /* Get the latest stat info from server. */
struct p9_fid *fid; struct p9_fid *fid;
fid = v9fs_fid_lookup(old_dentry); fid = v9fs_fid_lookup(old_dentry);
if (IS_ERR(fid)) if (IS_ERR(fid))
return PTR_ERR(fid); return PTR_ERR(fid);
...@@ -843,7 +842,7 @@ v9fs_vfs_mknod_dotl(struct user_namespace *mnt_userns, struct inode *dir, ...@@ -843,7 +842,7 @@ v9fs_vfs_mknod_dotl(struct user_namespace *mnt_userns, struct inode *dir,
struct p9_qid qid; struct p9_qid qid;
struct posix_acl *dacl = NULL, *pacl = NULL; struct posix_acl *dacl = NULL, *pacl = NULL;
p9_debug(P9_DEBUG_VFS, " %lu,%pd mode: %hx MAJOR: %u MINOR: %u\n", p9_debug(P9_DEBUG_VFS, " %lu,%pd mode: %x MAJOR: %u MINOR: %u\n",
dir->i_ino, dentry, omode, dir->i_ino, dentry, omode,
MAJOR(rdev), MINOR(rdev)); MAJOR(rdev), MINOR(rdev));
......
// SPDX-License-Identifier: GPL-2.0-only // SPDX-License-Identifier: GPL-2.0-only
/* /*
* linux/fs/9p/vfs_super.c
*
* This file contians superblock ops for 9P2000. It is intended that
* you mount this file system on directories.
* *
* Copyright (C) 2004 by Eric Van Hensbergen <ericvh@gmail.com> * Copyright (C) 2004 by Eric Van Hensbergen <ericvh@gmail.com>
* Copyright (C) 2002 by Ron Minnich <rminnich@lanl.gov> * Copyright (C) 2002 by Ron Minnich <rminnich@lanl.gov>
...@@ -83,6 +79,9 @@ v9fs_fill_super(struct super_block *sb, struct v9fs_session_info *v9ses, ...@@ -83,6 +79,9 @@ v9fs_fill_super(struct super_block *sb, struct v9fs_session_info *v9ses,
if (!v9ses->cache) { if (!v9ses->cache) {
sb->s_bdi->ra_pages = 0; sb->s_bdi->ra_pages = 0;
sb->s_bdi->io_pages = 0; sb->s_bdi->io_pages = 0;
} else {
sb->s_bdi->ra_pages = v9ses->maxdata >> PAGE_SHIFT;
sb->s_bdi->io_pages = v9ses->maxdata >> PAGE_SHIFT;
} }
sb->s_flags |= SB_ACTIVE | SB_DIRSYNC; sb->s_flags |= SB_ACTIVE | SB_DIRSYNC;
...@@ -113,7 +112,7 @@ static struct dentry *v9fs_mount(struct file_system_type *fs_type, int flags, ...@@ -113,7 +112,7 @@ static struct dentry *v9fs_mount(struct file_system_type *fs_type, int flags,
struct inode *inode = NULL; struct inode *inode = NULL;
struct dentry *root = NULL; struct dentry *root = NULL;
struct v9fs_session_info *v9ses = NULL; struct v9fs_session_info *v9ses = NULL;
umode_t mode = S_IRWXUGO | S_ISVTX; umode_t mode = 0777 | S_ISVTX;
struct p9_fid *fid; struct p9_fid *fid;
int retval = 0; int retval = 0;
...@@ -157,6 +156,7 @@ static struct dentry *v9fs_mount(struct file_system_type *fs_type, int flags, ...@@ -157,6 +156,7 @@ static struct dentry *v9fs_mount(struct file_system_type *fs_type, int flags,
sb->s_root = root; sb->s_root = root;
if (v9fs_proto_dotl(v9ses)) { if (v9fs_proto_dotl(v9ses)) {
struct p9_stat_dotl *st = NULL; struct p9_stat_dotl *st = NULL;
st = p9_client_getattr_dotl(fid, P9_STATS_BASIC); st = p9_client_getattr_dotl(fid, P9_STATS_BASIC);
if (IS_ERR(st)) { if (IS_ERR(st)) {
retval = PTR_ERR(st); retval = PTR_ERR(st);
...@@ -167,6 +167,7 @@ static struct dentry *v9fs_mount(struct file_system_type *fs_type, int flags, ...@@ -167,6 +167,7 @@ static struct dentry *v9fs_mount(struct file_system_type *fs_type, int flags,
kfree(st); kfree(st);
} else { } else {
struct p9_wstat *st = NULL; struct p9_wstat *st = NULL;
st = p9_client_stat(fid); st = p9_client_stat(fid);
if (IS_ERR(st)) { if (IS_ERR(st)) {
retval = PTR_ERR(st); retval = PTR_ERR(st);
...@@ -275,12 +276,13 @@ static int v9fs_statfs(struct dentry *dentry, struct kstatfs *buf) ...@@ -275,12 +276,13 @@ static int v9fs_statfs(struct dentry *dentry, struct kstatfs *buf)
static int v9fs_drop_inode(struct inode *inode) static int v9fs_drop_inode(struct inode *inode)
{ {
struct v9fs_session_info *v9ses; struct v9fs_session_info *v9ses;
v9ses = v9fs_inode2v9ses(inode); v9ses = v9fs_inode2v9ses(inode);
if (v9ses->cache == CACHE_LOOSE || v9ses->cache == CACHE_FSCACHE) if (v9ses->cache == CACHE_LOOSE || v9ses->cache == CACHE_FSCACHE)
return generic_drop_inode(inode); return generic_drop_inode(inode);
/* /*
* in case of non cached mode always drop the * in case of non cached mode always drop the
* the inode because we want the inode attribute * inode because we want the inode attribute
* to always match that on the server. * to always match that on the server.
*/ */
return 1; return 1;
......
// SPDX-License-Identifier: LGPL-2.1
/* /*
* Copyright IBM Corporation, 2010 * Copyright IBM Corporation, 2010
* Author Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com> * Author Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of version 2.1 of the GNU Lesser General Public License
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it would be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
*
*/ */
#include <linux/module.h> #include <linux/module.h>
......
/* SPDX-License-Identifier: LGPL-2.1 */
/* /*
* Copyright IBM Corporation, 2010 * Copyright IBM Corporation, 2010
* Author Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com> * Author Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of version 2.1 of the GNU Lesser General Public License
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it would be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
*
*/ */
#ifndef FS_9P_XATTR_H #ifndef FS_9P_XATTR_H
#define FS_9P_XATTR_H #define FS_9P_XATTR_H
...@@ -22,13 +14,14 @@ extern const struct xattr_handler *v9fs_xattr_handlers[]; ...@@ -22,13 +14,14 @@ extern const struct xattr_handler *v9fs_xattr_handlers[];
extern const struct xattr_handler v9fs_xattr_acl_access_handler; extern const struct xattr_handler v9fs_xattr_acl_access_handler;
extern const struct xattr_handler v9fs_xattr_acl_default_handler; extern const struct xattr_handler v9fs_xattr_acl_default_handler;
extern ssize_t v9fs_fid_xattr_get(struct p9_fid *, const char *, ssize_t v9fs_fid_xattr_get(struct p9_fid *fid, const char *name,
void *, size_t); void *buffer, size_t buffer_size);
extern ssize_t v9fs_xattr_get(struct dentry *, const char *, ssize_t v9fs_xattr_get(struct dentry *dentry, const char *name,
void *, size_t); void *buffer, size_t buffer_size);
extern int v9fs_fid_xattr_set(struct p9_fid *, const char *, int v9fs_fid_xattr_set(struct p9_fid *fid, const char *name,
const void *, size_t, int); const void *value, size_t value_len, int flags);
extern int v9fs_xattr_set(struct dentry *, const char *, int v9fs_xattr_set(struct dentry *dentry, const char *name,
const void *, size_t, int); const void *value, size_t value_len, int flags);
extern ssize_t v9fs_listxattr(struct dentry *, char *, size_t); ssize_t v9fs_listxattr(struct dentry *dentry, char *buffer,
size_t buffer_size);
#endif /* FS_9P_XATTR_H */ #endif /* FS_9P_XATTR_H */
...@@ -167,7 +167,7 @@ struct fscache_cookie { ...@@ -167,7 +167,7 @@ struct fscache_cookie {
static inline bool fscache_cookie_enabled(struct fscache_cookie *cookie) static inline bool fscache_cookie_enabled(struct fscache_cookie *cookie)
{ {
return test_bit(FSCACHE_COOKIE_ENABLED, &cookie->flags); return fscache_cookie_valid(cookie) && test_bit(FSCACHE_COOKIE_ENABLED, &cookie->flags);
} }
/* /*
......
/* SPDX-License-Identifier: GPL-2.0-only */ /* SPDX-License-Identifier: GPL-2.0-only */
/* /*
* include/net/9p/9p.h
*
* 9P protocol definitions. * 9P protocol definitions.
* *
* Copyright (C) 2005 by Latchesar Ionkov <lucho@ionkov.net> * Copyright (C) 2005 by Latchesar Ionkov <lucho@ionkov.net>
...@@ -32,13 +30,13 @@ ...@@ -32,13 +30,13 @@
*/ */
enum p9_debug_flags { enum p9_debug_flags {
P9_DEBUG_ERROR = (1<<0), P9_DEBUG_ERROR = (1<<0),
P9_DEBUG_9P = (1<<2), P9_DEBUG_9P = (1<<2),
P9_DEBUG_VFS = (1<<3), P9_DEBUG_VFS = (1<<3),
P9_DEBUG_CONV = (1<<4), P9_DEBUG_CONV = (1<<4),
P9_DEBUG_MUX = (1<<5), P9_DEBUG_MUX = (1<<5),
P9_DEBUG_TRANS = (1<<6), P9_DEBUG_TRANS = (1<<6),
P9_DEBUG_SLABS = (1<<7), P9_DEBUG_SLABS = (1<<7),
P9_DEBUG_FCALL = (1<<8), P9_DEBUG_FCALL = (1<<8),
P9_DEBUG_FID = (1<<9), P9_DEBUG_FID = (1<<9),
P9_DEBUG_PKT = (1<<10), P9_DEBUG_PKT = (1<<10),
...@@ -317,8 +315,8 @@ enum p9_qid_t { ...@@ -317,8 +315,8 @@ enum p9_qid_t {
}; };
/* 9P Magic Numbers */ /* 9P Magic Numbers */
#define P9_NOTAG (u16)(~0) #define P9_NOTAG ((u16)(~0))
#define P9_NOFID (u32)(~0) #define P9_NOFID ((u32)(~0))
#define P9_MAXWELEM 16 #define P9_MAXWELEM 16
/* Minimal header size: size[4] type[1] tag[2] */ /* Minimal header size: size[4] type[1] tag[2] */
......
/* SPDX-License-Identifier: GPL-2.0-only */ /* SPDX-License-Identifier: GPL-2.0-only */
/* /*
* include/net/9p/client.h
*
* 9P Client Definitions * 9P Client Definitions
* *
* Copyright (C) 2008 by Eric Van Hensbergen <ericvh@gmail.com> * Copyright (C) 2008 by Eric Van Hensbergen <ericvh@gmail.com>
...@@ -23,7 +21,7 @@ ...@@ -23,7 +21,7 @@
* @p9_proto_2000L: 9P2000.L extension * @p9_proto_2000L: 9P2000.L extension
*/ */
enum p9_proto_versions{ enum p9_proto_versions {
p9_proto_legacy, p9_proto_legacy,
p9_proto_2000u, p9_proto_2000u,
p9_proto_2000L, p9_proto_2000L,
...@@ -219,13 +217,13 @@ struct p9_stat_dotl *p9_client_getattr_dotl(struct p9_fid *fid, ...@@ -219,13 +217,13 @@ struct p9_stat_dotl *p9_client_getattr_dotl(struct p9_fid *fid,
u64 request_mask); u64 request_mask);
int p9_client_mknod_dotl(struct p9_fid *oldfid, const char *name, int mode, int p9_client_mknod_dotl(struct p9_fid *oldfid, const char *name, int mode,
dev_t rdev, kgid_t gid, struct p9_qid *); dev_t rdev, kgid_t gid, struct p9_qid *qid);
int p9_client_mkdir_dotl(struct p9_fid *fid, const char *name, int mode, int p9_client_mkdir_dotl(struct p9_fid *fid, const char *name, int mode,
kgid_t gid, struct p9_qid *); kgid_t gid, struct p9_qid *qid);
int p9_client_lock_dotl(struct p9_fid *fid, struct p9_flock *flock, u8 *status); int p9_client_lock_dotl(struct p9_fid *fid, struct p9_flock *flock, u8 *status);
int p9_client_getlock_dotl(struct p9_fid *fid, struct p9_getlock *fl); int p9_client_getlock_dotl(struct p9_fid *fid, struct p9_getlock *fl);
void p9_fcall_fini(struct p9_fcall *fc); void p9_fcall_fini(struct p9_fcall *fc);
struct p9_req_t *p9_tag_lookup(struct p9_client *, u16); struct p9_req_t *p9_tag_lookup(struct p9_client *c, u16 tag);
static inline void p9_req_get(struct p9_req_t *r) static inline void p9_req_get(struct p9_req_t *r)
{ {
...@@ -241,14 +239,18 @@ int p9_req_put(struct p9_req_t *r); ...@@ -241,14 +239,18 @@ int p9_req_put(struct p9_req_t *r);
void p9_client_cb(struct p9_client *c, struct p9_req_t *req, int status); void p9_client_cb(struct p9_client *c, struct p9_req_t *req, int status);
int p9_parse_header(struct p9_fcall *, int32_t *, int8_t *, int16_t *, int); int p9_parse_header(struct p9_fcall *pdu, int32_t *size, int8_t *type,
int p9stat_read(struct p9_client *, char *, int, struct p9_wstat *); int16_t *tag, int rewind);
void p9stat_free(struct p9_wstat *); int p9stat_read(struct p9_client *clnt, char *buf, int len,
struct p9_wstat *st);
void p9stat_free(struct p9_wstat *stbuf);
int p9_is_proto_dotu(struct p9_client *clnt); int p9_is_proto_dotu(struct p9_client *clnt);
int p9_is_proto_dotl(struct p9_client *clnt); int p9_is_proto_dotl(struct p9_client *clnt);
struct p9_fid *p9_client_xattrwalk(struct p9_fid *, const char *, u64 *); struct p9_fid *p9_client_xattrwalk(struct p9_fid *file_fid,
int p9_client_xattrcreate(struct p9_fid *, const char *, u64, int); const char *attr_name, u64 *attr_size);
int p9_client_xattrcreate(struct p9_fid *fid, const char *name,
u64 attr_size, int flags);
int p9_client_readlink(struct p9_fid *fid, char **target); int p9_client_readlink(struct p9_fid *fid, char **target);
int p9_client_init(void); int p9_client_init(void);
......
/* SPDX-License-Identifier: GPL-2.0-only */ /* SPDX-License-Identifier: GPL-2.0-only */
/* /*
* include/net/9p/transport.h
*
* Transport Definition * Transport Definition
* *
* Copyright (C) 2005 by Latchesar Ionkov <lucho@ionkov.net> * Copyright (C) 2005 by Latchesar Ionkov <lucho@ionkov.net>
...@@ -11,6 +9,8 @@ ...@@ -11,6 +9,8 @@
#ifndef NET_9P_TRANSPORT_H #ifndef NET_9P_TRANSPORT_H
#define NET_9P_TRANSPORT_H #define NET_9P_TRANSPORT_H
#include <linux/module.h>
#define P9_DEF_MIN_RESVPORT (665U) #define P9_DEF_MIN_RESVPORT (665U)
#define P9_DEF_MAX_RESVPORT (1023U) #define P9_DEF_MAX_RESVPORT (1023U)
...@@ -40,14 +40,16 @@ struct p9_trans_module { ...@@ -40,14 +40,16 @@ struct p9_trans_module {
int maxsize; /* max message size of transport */ int maxsize; /* max message size of transport */
int def; /* this transport should be default */ int def; /* this transport should be default */
struct module *owner; struct module *owner;
int (*create)(struct p9_client *, const char *, char *); int (*create)(struct p9_client *client,
void (*close) (struct p9_client *); const char *devname, char *args);
int (*request) (struct p9_client *, struct p9_req_t *req); void (*close)(struct p9_client *client);
int (*cancel) (struct p9_client *, struct p9_req_t *req); int (*request)(struct p9_client *client, struct p9_req_t *req);
int (*cancelled)(struct p9_client *, struct p9_req_t *req); int (*cancel)(struct p9_client *client, struct p9_req_t *req);
int (*zc_request)(struct p9_client *, struct p9_req_t *, int (*cancelled)(struct p9_client *client, struct p9_req_t *req);
struct iov_iter *, struct iov_iter *, int , int, int); int (*zc_request)(struct p9_client *client, struct p9_req_t *req,
int (*show_options)(struct seq_file *, struct p9_client *); struct iov_iter *uidata, struct iov_iter *uodata,
int inlen, int outlen, int in_hdr_len);
int (*show_options)(struct seq_file *m, struct p9_client *client);
}; };
void v9fs_register_trans(struct p9_trans_module *m); void v9fs_register_trans(struct p9_trans_module *m);
...@@ -55,4 +57,8 @@ void v9fs_unregister_trans(struct p9_trans_module *m); ...@@ -55,4 +57,8 @@ void v9fs_unregister_trans(struct p9_trans_module *m);
struct p9_trans_module *v9fs_get_trans_by_name(char *s); struct p9_trans_module *v9fs_get_trans_by_name(char *s);
struct p9_trans_module *v9fs_get_default_trans(void); struct p9_trans_module *v9fs_get_default_trans(void);
void v9fs_put_trans(struct p9_trans_module *m); void v9fs_put_trans(struct p9_trans_module *m);
#define MODULE_ALIAS_9P(transport) \
MODULE_ALIAS("9p-" transport)
#endif /* NET_9P_TRANSPORT_H */ #endif /* NET_9P_TRANSPORT_H */
This diff is collapsed.
// SPDX-License-Identifier: GPL-2.0-only // SPDX-License-Identifier: GPL-2.0-only
/* /*
* linux/fs/9p/error.c
*
* Error string handling * Error string handling
* *
* Plan 9 uses error strings, Unix uses error numbers. These functions * Plan 9 uses error strings, Unix uses error numbers. These functions
...@@ -185,7 +183,7 @@ int p9_error_init(void) ...@@ -185,7 +183,7 @@ int p9_error_init(void)
INIT_HLIST_HEAD(&hash_errmap[bucket]); INIT_HLIST_HEAD(&hash_errmap[bucket]);
/* load initial error map into hash table */ /* load initial error map into hash table */
for (c = errmap; c->name != NULL; c++) { for (c = errmap; c->name; c++) {
c->namelen = strlen(c->name); c->namelen = strlen(c->name);
bucket = jhash(c->name, c->namelen, 0) % ERRHASHSZ; bucket = jhash(c->name, c->namelen, 0) % ERRHASHSZ;
INIT_HLIST_NODE(&c->list); INIT_HLIST_NODE(&c->list);
......
// SPDX-License-Identifier: GPL-2.0-only // SPDX-License-Identifier: GPL-2.0-only
/* /*
* net/9p/9p.c
*
* 9P entry point * 9P entry point
* *
* Copyright (C) 2007 by Latchesar Ionkov <lucho@ionkov.net> * Copyright (C) 2007 by Latchesar Ionkov <lucho@ionkov.net>
...@@ -12,6 +10,7 @@ ...@@ -12,6 +10,7 @@
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
#include <linux/module.h> #include <linux/module.h>
#include <linux/kmod.h>
#include <linux/errno.h> #include <linux/errno.h>
#include <linux/sched.h> #include <linux/sched.h>
#include <linux/moduleparam.h> #include <linux/moduleparam.h>
...@@ -24,13 +23,13 @@ ...@@ -24,13 +23,13 @@
#include <linux/spinlock.h> #include <linux/spinlock.h>
#ifdef CONFIG_NET_9P_DEBUG #ifdef CONFIG_NET_9P_DEBUG
unsigned int p9_debug_level = 0; /* feature-rific global debug level */ unsigned int p9_debug_level; /* feature-rific global debug level */
EXPORT_SYMBOL(p9_debug_level); EXPORT_SYMBOL(p9_debug_level);
module_param_named(debug, p9_debug_level, uint, 0); module_param_named(debug, p9_debug_level, uint, 0);
MODULE_PARM_DESC(debug, "9P debugging level"); MODULE_PARM_DESC(debug, "9P debugging level");
void _p9_debug(enum p9_debug_flags level, const char *func, void _p9_debug(enum p9_debug_flags level, const char *func,
const char *fmt, ...) const char *fmt, ...)
{ {
struct va_format vaf; struct va_format vaf;
va_list args; va_list args;
...@@ -53,10 +52,7 @@ void _p9_debug(enum p9_debug_flags level, const char *func, ...@@ -53,10 +52,7 @@ void _p9_debug(enum p9_debug_flags level, const char *func,
EXPORT_SYMBOL(_p9_debug); EXPORT_SYMBOL(_p9_debug);
#endif #endif
/* /* Dynamic Transport Registration Routines */
* Dynamic Transport Registration Routines
*
*/
static DEFINE_SPINLOCK(v9fs_trans_lock); static DEFINE_SPINLOCK(v9fs_trans_lock);
static LIST_HEAD(v9fs_trans_list); static LIST_HEAD(v9fs_trans_list);
...@@ -87,12 +83,7 @@ void v9fs_unregister_trans(struct p9_trans_module *m) ...@@ -87,12 +83,7 @@ void v9fs_unregister_trans(struct p9_trans_module *m)
} }
EXPORT_SYMBOL(v9fs_unregister_trans); EXPORT_SYMBOL(v9fs_unregister_trans);
/** static struct p9_trans_module *_p9_get_trans_by_name(char *s)
* v9fs_get_trans_by_name - get transport with the matching name
* @s: string identifying transport
*
*/
struct p9_trans_module *v9fs_get_trans_by_name(char *s)
{ {
struct p9_trans_module *t, *found = NULL; struct p9_trans_module *t, *found = NULL;
...@@ -106,6 +97,28 @@ struct p9_trans_module *v9fs_get_trans_by_name(char *s) ...@@ -106,6 +97,28 @@ struct p9_trans_module *v9fs_get_trans_by_name(char *s)
} }
spin_unlock(&v9fs_trans_lock); spin_unlock(&v9fs_trans_lock);
return found;
}
/**
* v9fs_get_trans_by_name - get transport with the matching name
* @s: string identifying transport
*
*/
struct p9_trans_module *v9fs_get_trans_by_name(char *s)
{
struct p9_trans_module *found = NULL;
found = _p9_get_trans_by_name(s);
#ifdef CONFIG_MODULES
if (!found) {
request_module("9p-%s", s);
found = _p9_get_trans_by_name(s);
}
#endif
return found; return found;
} }
EXPORT_SYMBOL(v9fs_get_trans_by_name); EXPORT_SYMBOL(v9fs_get_trans_by_name);
......
// SPDX-License-Identifier: GPL-2.0-only // SPDX-License-Identifier: GPL-2.0-only
/* /*
* net/9p/protocol.c
*
* 9P Protocol Support Code * 9P Protocol Support Code
* *
* Copyright (C) 2008 by Eric Van Hensbergen <ericvh@gmail.com> * Copyright (C) 2008 by Eric Van Hensbergen <ericvh@gmail.com>
...@@ -46,6 +44,7 @@ EXPORT_SYMBOL(p9stat_free); ...@@ -46,6 +44,7 @@ EXPORT_SYMBOL(p9stat_free);
size_t pdu_read(struct p9_fcall *pdu, void *data, size_t size) size_t pdu_read(struct p9_fcall *pdu, void *data, size_t size)
{ {
size_t len = min(pdu->size - pdu->offset, size); size_t len = min(pdu->size - pdu->offset, size);
memcpy(data, &pdu->sdata[pdu->offset], len); memcpy(data, &pdu->sdata[pdu->offset], len);
pdu->offset += len; pdu->offset += len;
return size - len; return size - len;
...@@ -54,6 +53,7 @@ size_t pdu_read(struct p9_fcall *pdu, void *data, size_t size) ...@@ -54,6 +53,7 @@ size_t pdu_read(struct p9_fcall *pdu, void *data, size_t size)
static size_t pdu_write(struct p9_fcall *pdu, const void *data, size_t size) static size_t pdu_write(struct p9_fcall *pdu, const void *data, size_t size)
{ {
size_t len = min(pdu->capacity - pdu->size, size); size_t len = min(pdu->capacity - pdu->size, size);
memcpy(&pdu->sdata[pdu->size], data, len); memcpy(&pdu->sdata[pdu->size], data, len);
pdu->size += len; pdu->size += len;
return size - len; return size - len;
...@@ -64,6 +64,7 @@ pdu_write_u(struct p9_fcall *pdu, struct iov_iter *from, size_t size) ...@@ -64,6 +64,7 @@ pdu_write_u(struct p9_fcall *pdu, struct iov_iter *from, size_t size)
{ {
size_t len = min(pdu->capacity - pdu->size, size); size_t len = min(pdu->capacity - pdu->size, size);
struct iov_iter i = *from; struct iov_iter i = *from;
if (!copy_from_iter_full(&pdu->sdata[pdu->size], len, &i)) if (!copy_from_iter_full(&pdu->sdata[pdu->size], len, &i))
len = 0; len = 0;
...@@ -71,26 +72,25 @@ pdu_write_u(struct p9_fcall *pdu, struct iov_iter *from, size_t size) ...@@ -71,26 +72,25 @@ pdu_write_u(struct p9_fcall *pdu, struct iov_iter *from, size_t size)
return size - len; return size - len;
} }
/* /* b - int8_t
b - int8_t * w - int16_t
w - int16_t * d - int32_t
d - int32_t * q - int64_t
q - int64_t * s - string
s - string * u - numeric uid
u - numeric uid * g - numeric gid
g - numeric gid * S - stat
S - stat * Q - qid
Q - qid * D - data blob (int32_t size followed by void *, results are not freed)
D - data blob (int32_t size followed by void *, results are not freed) * T - array of strings (int16_t count, followed by strings)
T - array of strings (int16_t count, followed by strings) * R - array of qids (int16_t count, followed by qids)
R - array of qids (int16_t count, followed by qids) * A - stat for 9p2000.L (p9_stat_dotl)
A - stat for 9p2000.L (p9_stat_dotl) * ? - if optional = 1, continue parsing
? - if optional = 1, continue parsing */
*/
static int static int
p9pdu_vreadf(struct p9_fcall *pdu, int proto_version, const char *fmt, p9pdu_vreadf(struct p9_fcall *pdu, int proto_version, const char *fmt,
va_list ap) va_list ap)
{ {
const char *ptr; const char *ptr;
int errcode = 0; int errcode = 0;
......
/* SPDX-License-Identifier: GPL-2.0-only */ /* SPDX-License-Identifier: GPL-2.0-only */
/* /*
* net/9p/protocol.h
*
* 9P Protocol Support Code * 9P Protocol Support Code
* *
* Copyright (C) 2008 by Eric Van Hensbergen <ericvh@gmail.com> * Copyright (C) 2008 by Eric Van Hensbergen <ericvh@gmail.com>
...@@ -11,7 +9,7 @@ ...@@ -11,7 +9,7 @@
*/ */
int p9pdu_vwritef(struct p9_fcall *pdu, int proto_version, const char *fmt, int p9pdu_vwritef(struct p9_fcall *pdu, int proto_version, const char *fmt,
va_list ap); va_list ap);
int p9pdu_readf(struct p9_fcall *pdu, int proto_version, const char *fmt, ...); int p9pdu_readf(struct p9_fcall *pdu, int proto_version, const char *fmt, ...);
int p9pdu_prepare(struct p9_fcall *pdu, int16_t tag, int8_t type); int p9pdu_prepare(struct p9_fcall *pdu, int16_t tag, int8_t type);
int p9pdu_finalize(struct p9_client *clnt, struct p9_fcall *pdu); int p9pdu_finalize(struct p9_client *clnt, struct p9_fcall *pdu);
......
// SPDX-License-Identifier: LGPL-2.1
/* /*
* Copyright IBM Corporation, 2010 * Copyright IBM Corporation, 2010
* Author Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com> * Author Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of version 2.1 of the GNU Lesser General Public License
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it would be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
*
*/ */
#include <linux/mm.h> #include <linux/mm.h>
......
/* SPDX-License-Identifier: LGPL-2.1 */
/* /*
* Copyright IBM Corporation, 2010 * Copyright IBM Corporation, 2010
* Author Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com> * Author Venkateswararao Jujjuri <jvrao@linux.vnet.ibm.com>
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of version 2.1 of the GNU Lesser General Public License
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it would be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
*
*/ */
void p9_release_pages(struct page **, int); void p9_release_pages(struct page **pages, int nr_pages);
// SPDX-License-Identifier: GPL-2.0-only // SPDX-License-Identifier: GPL-2.0-only
/* /*
* linux/fs/9p/trans_fd.c
*
* Fd transport layer. Includes deprecated socket layer. * Fd transport layer. Includes deprecated socket layer.
* *
* Copyright (C) 2006 by Russ Cox <rsc@swtch.com> * Copyright (C) 2006 by Russ Cox <rsc@swtch.com>
......
// SPDX-License-Identifier: GPL-2.0-only // SPDX-License-Identifier: GPL-2.0-only
/* /*
* linux/fs/9p/trans_rdma.c
*
* RDMA transport layer based on the trans_fd.c implementation. * RDMA transport layer based on the trans_fd.c implementation.
* *
* Copyright (C) 2008 by Tom Tucker <tom@opengridcomputing.com> * Copyright (C) 2008 by Tom Tucker <tom@opengridcomputing.com>
...@@ -767,6 +765,7 @@ static void __exit p9_trans_rdma_exit(void) ...@@ -767,6 +765,7 @@ static void __exit p9_trans_rdma_exit(void)
module_init(p9_trans_rdma_init); module_init(p9_trans_rdma_init);
module_exit(p9_trans_rdma_exit); module_exit(p9_trans_rdma_exit);
MODULE_ALIAS_9P("rdma");
MODULE_AUTHOR("Tom Tucker <tom@opengridcomputing.com>"); MODULE_AUTHOR("Tom Tucker <tom@opengridcomputing.com>");
MODULE_DESCRIPTION("RDMA Transport for 9P"); MODULE_DESCRIPTION("RDMA Transport for 9P");
......
...@@ -794,6 +794,7 @@ static void __exit p9_virtio_cleanup(void) ...@@ -794,6 +794,7 @@ static void __exit p9_virtio_cleanup(void)
module_init(p9_virtio_init); module_init(p9_virtio_init);
module_exit(p9_virtio_cleanup); module_exit(p9_virtio_cleanup);
MODULE_ALIAS_9P("virtio");
MODULE_DEVICE_TABLE(virtio, id_table); MODULE_DEVICE_TABLE(virtio, id_table);
MODULE_AUTHOR("Eric Van Hensbergen <ericvh@gmail.com>"); MODULE_AUTHOR("Eric Van Hensbergen <ericvh@gmail.com>");
......
// SPDX-License-Identifier: GPL-2.0-only
/* /*
* linux/fs/9p/trans_xen * linux/fs/9p/trans_xen
* *
* Xen transport layer. * Xen transport layer.
* *
* Copyright (C) 2017 by Stefano Stabellini <stefano@aporeto.com> * Copyright (C) 2017 by Stefano Stabellini <stefano@aporeto.com>
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License version 2
* as published by the Free Software Foundation; or, when distributed
* separately from the Linux kernel or incorporated into other
* software packages, subject to the following license:
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this source file (the "Software"), to deal in the Software without
* restriction, including without limitation the rights to use, copy, modify,
* merge, publish, distribute, sublicense, and/or sell copies of the Software,
* and to permit persons to whom the Software is furnished to do so, subject to
* the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
* IN THE SOFTWARE.
*/ */
#include <xen/events.h> #include <xen/events.h>
...@@ -552,6 +529,7 @@ static int p9_trans_xen_init(void) ...@@ -552,6 +529,7 @@ static int p9_trans_xen_init(void)
return rc; return rc;
} }
module_init(p9_trans_xen_init); module_init(p9_trans_xen_init);
MODULE_ALIAS_9P("xen");
static void p9_trans_xen_exit(void) static void p9_trans_xen_exit(void)
{ {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment