Commit 2874b391 authored by Linus Torvalds's avatar Linus Torvalds

Merge branch 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/ericvh/v9fs

* 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/ericvh/v9fs:
  9p: implement optional loose read cache
  9p: Use kthread_stop instead of sending a SIGKILL.
parents 5fc77247 e03abc0c
...@@ -4,6 +4,8 @@ Exporting ...@@ -4,6 +4,8 @@ Exporting
- explanation of how to make filesystems exportable. - explanation of how to make filesystems exportable.
Locking Locking
- info on locking rules as they pertain to Linux VFS. - info on locking rules as they pertain to Linux VFS.
9p.txt
- 9p (v9fs) is an implementation of the Plan 9 remote fs protocol.
adfs.txt adfs.txt
- info and mount options for the Acorn Advanced Disc Filing System. - info and mount options for the Acorn Advanced Disc Filing System.
afs.txt afs.txt
...@@ -82,8 +84,6 @@ udf.txt ...@@ -82,8 +84,6 @@ udf.txt
- info and mount options for the UDF filesystem. - info and mount options for the UDF filesystem.
ufs.txt ufs.txt
- info on the ufs filesystem. - info on the ufs filesystem.
v9fs.txt
- v9fs is a Unix implementation of the Plan 9 9p remote fs protocol.
vfat.txt vfat.txt
- info on using the VFAT filesystem used in Windows NT and Windows 95 - info on using the VFAT filesystem used in Windows NT and Windows 95
vfs.txt vfs.txt
......
...@@ -40,6 +40,10 @@ OPTIONS ...@@ -40,6 +40,10 @@ OPTIONS
aname=name aname specifies the file tree to access when the server is aname=name aname specifies the file tree to access when the server is
offering several exported file systems. offering several exported file systems.
cache=mode specifies a cacheing policy. By default, no caches are used.
loose = no attempts are made at consistency,
intended for exclusive, read-only mounts
debug=n specifies debug level. The debug level is a bitmask. debug=n specifies debug level. The debug level is a bitmask.
0x01 = display verbose error messages 0x01 = display verbose error messages
0x02 = developer debug (DEBUG_CURRENT) 0x02 = developer debug (DEBUG_CURRENT)
......
...@@ -136,7 +136,8 @@ struct v9fs_fid *v9fs_fid_lookup(struct dentry *dentry) ...@@ -136,7 +136,8 @@ struct v9fs_fid *v9fs_fid_lookup(struct dentry *dentry)
} }
/** /**
* v9fs_fid_clone - lookup the fid for a dentry, clone a private copy and release it * v9fs_fid_clone - lookup the fid for a dentry, clone a private copy and
* release it
* @dentry: dentry to look for fid in * @dentry: dentry to look for fid in
* *
* find a fid in the dentry and then clone to a new private fid * find a fid in the dentry and then clone to a new private fid
......
...@@ -256,7 +256,7 @@ static void v9fs_mux_poll_stop(struct v9fs_mux_data *m) ...@@ -256,7 +256,7 @@ static void v9fs_mux_poll_stop(struct v9fs_mux_data *m)
vpt->muxnum--; vpt->muxnum--;
if (!vpt->muxnum) { if (!vpt->muxnum) {
dprintk(DEBUG_MUX, "destroy proc %p\n", vpt); dprintk(DEBUG_MUX, "destroy proc %p\n", vpt);
send_sig(SIGKILL, vpt->task, 1); kthread_stop(vpt->task);
vpt->task = NULL; vpt->task = NULL;
v9fs_mux_poll_task_num--; v9fs_mux_poll_task_num--;
} }
...@@ -438,11 +438,8 @@ static int v9fs_poll_proc(void *a) ...@@ -438,11 +438,8 @@ static int v9fs_poll_proc(void *a)
vpt = a; vpt = a;
dprintk(DEBUG_MUX, "start %p %p\n", current, vpt); dprintk(DEBUG_MUX, "start %p %p\n", current, vpt);
allow_signal(SIGKILL);
while (!kthread_should_stop()) { while (!kthread_should_stop()) {
set_current_state(TASK_INTERRUPTIBLE); set_current_state(TASK_INTERRUPTIBLE);
if (signal_pending(current))
break;
list_for_each_entry_safe(m, mtmp, &vpt->mux_list, mux_list) { list_for_each_entry_safe(m, mtmp, &vpt->mux_list, mux_list) {
v9fs_poll_mux(m); v9fs_poll_mux(m);
......
...@@ -53,6 +53,8 @@ enum { ...@@ -53,6 +53,8 @@ enum {
Opt_uname, Opt_remotename, Opt_uname, Opt_remotename,
/* Options that take no arguments */ /* Options that take no arguments */
Opt_legacy, Opt_nodevmap, Opt_unix, Opt_tcp, Opt_fd, Opt_legacy, Opt_nodevmap, Opt_unix, Opt_tcp, Opt_fd,
/* Cache options */
Opt_cache_loose,
/* Error token */ /* Error token */
Opt_err Opt_err
}; };
...@@ -76,6 +78,8 @@ static match_table_t tokens = { ...@@ -76,6 +78,8 @@ static match_table_t tokens = {
{Opt_fd, "fd"}, {Opt_fd, "fd"},
{Opt_legacy, "noextend"}, {Opt_legacy, "noextend"},
{Opt_nodevmap, "nodevmap"}, {Opt_nodevmap, "nodevmap"},
{Opt_cache_loose, "cache=loose"},
{Opt_cache_loose, "loose"},
{Opt_err, NULL} {Opt_err, NULL}
}; };
...@@ -106,6 +110,7 @@ static void v9fs_parse_options(char *options, struct v9fs_session_info *v9ses) ...@@ -106,6 +110,7 @@ static void v9fs_parse_options(char *options, struct v9fs_session_info *v9ses)
v9ses->debug = 0; v9ses->debug = 0;
v9ses->rfdno = ~0; v9ses->rfdno = ~0;
v9ses->wfdno = ~0; v9ses->wfdno = ~0;
v9ses->cache = 0;
if (!options) if (!options)
return; return;
...@@ -121,7 +126,6 @@ static void v9fs_parse_options(char *options, struct v9fs_session_info *v9ses) ...@@ -121,7 +126,6 @@ static void v9fs_parse_options(char *options, struct v9fs_session_info *v9ses)
"integer field, but no integer?\n"); "integer field, but no integer?\n");
continue; continue;
} }
} }
switch (token) { switch (token) {
case Opt_port: case Opt_port:
...@@ -169,6 +173,9 @@ static void v9fs_parse_options(char *options, struct v9fs_session_info *v9ses) ...@@ -169,6 +173,9 @@ static void v9fs_parse_options(char *options, struct v9fs_session_info *v9ses)
case Opt_nodevmap: case Opt_nodevmap:
v9ses->nodev = 1; v9ses->nodev = 1;
break; break;
case Opt_cache_loose:
v9ses->cache = CACHE_LOOSE;
break;
default: default:
continue; continue;
} }
......
...@@ -47,7 +47,7 @@ struct v9fs_session_info { ...@@ -47,7 +47,7 @@ struct v9fs_session_info {
unsigned int afid; /* authentication fid */ unsigned int afid; /* authentication fid */
unsigned int rfdno; /* read file descriptor number */ unsigned int rfdno; /* read file descriptor number */
unsigned int wfdno; /* write file descriptor number */ unsigned int wfdno; /* write file descriptor number */
unsigned int cache; /* cache mode */
char *name; /* user name to mount as */ char *name; /* user name to mount as */
char *remotename; /* name of remote hierarchy being mounted */ char *remotename; /* name of remote hierarchy being mounted */
...@@ -73,6 +73,13 @@ enum { ...@@ -73,6 +73,13 @@ enum {
PROTO_FD, PROTO_FD,
}; };
/* possible values of ->cache */
/* eventually support loose, tight, time, session, default always none */
enum {
CACHE_NONE, /* default */
CACHE_LOOSE, /* no consistency */
};
extern struct dentry *v9fs_debugfs_root; extern struct dentry *v9fs_debugfs_root;
int v9fs_session_init(struct v9fs_session_info *, const char *, char *); int v9fs_session_init(struct v9fs_session_info *, const char *, char *);
......
...@@ -40,8 +40,10 @@ ...@@ -40,8 +40,10 @@
extern struct file_system_type v9fs_fs_type; extern struct file_system_type v9fs_fs_type;
extern const struct address_space_operations v9fs_addr_operations; extern const struct address_space_operations v9fs_addr_operations;
extern const struct file_operations v9fs_file_operations; extern const struct file_operations v9fs_file_operations;
extern const struct file_operations v9fs_cached_file_operations;
extern const struct file_operations v9fs_dir_operations; extern const struct file_operations v9fs_dir_operations;
extern struct dentry_operations v9fs_dentry_operations; extern struct dentry_operations v9fs_dentry_operations;
extern struct dentry_operations v9fs_cached_dentry_operations;
struct inode *v9fs_get_inode(struct super_block *sb, int mode); struct inode *v9fs_get_inode(struct super_block *sb, int mode);
ino_t v9fs_qid2ino(struct v9fs_qid *qid); ino_t v9fs_qid2ino(struct v9fs_qid *qid);
......
...@@ -63,6 +63,8 @@ static int v9fs_vfs_readpage(struct file *filp, struct page *page) ...@@ -63,6 +63,8 @@ static int v9fs_vfs_readpage(struct file *filp, struct page *page)
int total = 0; int total = 0;
int result = 0; int result = 0;
dprintk(DEBUG_VFS, "\n");
buffer = kmap(page); buffer = kmap(page);
do { do {
if (count < rsize) if (count < rsize)
......
...@@ -53,9 +53,30 @@ ...@@ -53,9 +53,30 @@
static int v9fs_dentry_delete(struct dentry *dentry) static int v9fs_dentry_delete(struct dentry *dentry)
{ {
dprintk(DEBUG_VFS, " dentry: %s (%p)\n", dentry->d_iname, dentry); dprintk(DEBUG_VFS, " dentry: %s (%p)\n", dentry->d_iname, dentry);
return 1; return 1;
} }
/**
* v9fs_cached_dentry_delete - called when dentry refcount equals 0
* @dentry: dentry in question
*
* Only return 1 if our inode is invalid. Only non-synthetic files
* (ones without mtime == 0) should be calling this function.
*
*/
static int v9fs_cached_dentry_delete(struct dentry *dentry)
{
struct inode *inode = dentry->d_inode;
dprintk(DEBUG_VFS, " dentry: %s (%p)\n", dentry->d_iname, dentry);
if(!inode)
return 1;
return 0;
}
/** /**
* v9fs_dentry_release - called when dentry is going to be freed * v9fs_dentry_release - called when dentry is going to be freed
* @dentry: dentry that is being release * @dentry: dentry that is being release
...@@ -87,6 +108,11 @@ void v9fs_dentry_release(struct dentry *dentry) ...@@ -87,6 +108,11 @@ void v9fs_dentry_release(struct dentry *dentry)
} }
} }
struct dentry_operations v9fs_cached_dentry_operations = {
.d_delete = v9fs_cached_dentry_delete,
.d_release = v9fs_dentry_release,
};
struct dentry_operations v9fs_dentry_operations = { struct dentry_operations v9fs_dentry_operations = {
.d_delete = v9fs_dentry_delete, .d_delete = v9fs_dentry_delete,
.d_release = v9fs_dentry_release, .d_release = v9fs_dentry_release,
......
...@@ -79,6 +79,13 @@ int v9fs_file_open(struct inode *inode, struct file *file) ...@@ -79,6 +79,13 @@ int v9fs_file_open(struct inode *inode, struct file *file)
vfid->filp = file; vfid->filp = file;
kfree(fcall); kfree(fcall);
if((vfid->qid.version) && (v9ses->cache)) {
dprintk(DEBUG_VFS, "cached");
/* enable cached file options */
if(file->f_op == &v9fs_file_operations)
file->f_op = &v9fs_cached_file_operations;
}
return 0; return 0;
Clunk_Fid: Clunk_Fid:
...@@ -238,6 +245,17 @@ v9fs_file_write(struct file *filp, const char __user * data, ...@@ -238,6 +245,17 @@ v9fs_file_write(struct file *filp, const char __user * data,
return total; return total;
} }
const struct file_operations v9fs_cached_file_operations = {
.llseek = generic_file_llseek,
.read = do_sync_read,
.aio_read = generic_file_aio_read,
.write = v9fs_file_write,
.open = v9fs_file_open,
.release = v9fs_dir_release,
.lock = v9fs_file_lock,
.mmap = generic_file_mmap,
};
const struct file_operations v9fs_file_operations = { const struct file_operations v9fs_file_operations = {
.llseek = generic_file_llseek, .llseek = generic_file_llseek,
.read = v9fs_file_read, .read = v9fs_file_read,
......
...@@ -504,7 +504,10 @@ v9fs_vfs_create(struct inode *dir, struct dentry *dentry, int mode, ...@@ -504,7 +504,10 @@ v9fs_vfs_create(struct inode *dir, struct dentry *dentry, int mode,
goto error; goto error;
} }
dentry->d_op = &v9fs_dentry_operations; if(v9ses->cache)
dentry->d_op = &v9fs_cached_dentry_operations;
else
dentry->d_op = &v9fs_dentry_operations;
d_instantiate(dentry, inode); d_instantiate(dentry, inode);
if (nd && nd->flags & LOOKUP_OPEN) { if (nd && nd->flags & LOOKUP_OPEN) {
...@@ -589,7 +592,10 @@ static int v9fs_vfs_mkdir(struct inode *dir, struct dentry *dentry, int mode) ...@@ -589,7 +592,10 @@ static int v9fs_vfs_mkdir(struct inode *dir, struct dentry *dentry, int mode)
goto error; goto error;
} }
dentry->d_op = &v9fs_dentry_operations; if(v9ses->cache)
dentry->d_op = &v9fs_cached_dentry_operations;
else
dentry->d_op = &v9fs_dentry_operations;
d_instantiate(dentry, inode); d_instantiate(dentry, inode);
return 0; return 0;
...@@ -626,7 +632,6 @@ static struct dentry *v9fs_vfs_lookup(struct inode *dir, struct dentry *dentry, ...@@ -626,7 +632,6 @@ static struct dentry *v9fs_vfs_lookup(struct inode *dir, struct dentry *dentry,
sb = dir->i_sb; sb = dir->i_sb;
v9ses = v9fs_inode2v9ses(dir); v9ses = v9fs_inode2v9ses(dir);
dentry->d_op = &v9fs_dentry_operations;
dirfid = v9fs_fid_lookup(dentry->d_parent); dirfid = v9fs_fid_lookup(dentry->d_parent);
if(IS_ERR(dirfid)) if(IS_ERR(dirfid))
...@@ -697,6 +702,10 @@ static struct dentry *v9fs_vfs_lookup(struct inode *dir, struct dentry *dentry, ...@@ -697,6 +702,10 @@ static struct dentry *v9fs_vfs_lookup(struct inode *dir, struct dentry *dentry,
fid->qid = fcall->params.rstat.stat.qid; fid->qid = fcall->params.rstat.stat.qid;
v9fs_stat2inode(&fcall->params.rstat.stat, inode, inode->i_sb); v9fs_stat2inode(&fcall->params.rstat.stat, inode, inode->i_sb);
if((fid->qid.version)&&(v9ses->cache))
dentry->d_op = &v9fs_cached_dentry_operations;
else
dentry->d_op = &v9fs_dentry_operations;
d_add(dentry, inode); d_add(dentry, inode);
kfree(fcall); kfree(fcall);
...@@ -1184,7 +1193,10 @@ static int v9fs_vfs_mkspecial(struct inode *dir, struct dentry *dentry, ...@@ -1184,7 +1193,10 @@ static int v9fs_vfs_mkspecial(struct inode *dir, struct dentry *dentry,
goto free_vfid; goto free_vfid;
} }
dentry->d_op = &v9fs_dentry_operations; if(v9ses->cache)
dentry->d_op = &v9fs_cached_dentry_operations;
else
dentry->d_op = &v9fs_dentry_operations;
d_instantiate(dentry, inode); d_instantiate(dentry, inode);
return 0; return 0;
......
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