diff options
author | Miklos Szeredi <miklos@szeredi.hu> | 2006-10-17 03:10:06 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@g5.osdl.org> | 2006-10-17 11:18:45 -0400 |
commit | 9ffbb9162312fd8113037cb3d94f787f06bbfa9a (patch) | |
tree | a8194bb542dcda56271b44d7de23f2a72ecac900 /fs/fuse/inode.c | |
parent | 48d1a7ea6373337985f27dc1c707649469df5827 (diff) |
[PATCH] fuse: fix hang on SMP
Fuse didn't always call i_size_write() with i_mutex held which caused rare
hangs on SMP/32bit. This bug has been present since fuse-2.2, well before
being merged into mainline.
The simplest solution is to protect i_size_write() with the per-connection
spinlock. Using i_mutex for this purpose would require some restructuring of
the code and I'm not even sure it's always safe to acquire i_mutex in all
places i_size needs to be set.
Since most of vmtruncate is already duplicated for other reasons, duplicate
the remaining part as well, making all i_size_write() calls internal to fuse.
Using i_size_write() was unnecessary in fuse_init_inode(), since this function
is only called on a newly created locked inode.
Reported by a few people over the years, but special thanks to Dana Henriksen
who was persistent enough in helping me debug it.
Signed-off-by: Miklos Szeredi <miklos@szeredi.hu>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Diffstat (limited to 'fs/fuse/inode.c')
-rw-r--r-- | fs/fuse/inode.c | 5 |
1 files changed, 4 insertions, 1 deletions
diff --git a/fs/fuse/inode.c b/fs/fuse/inode.c index 7d0a9aee01f2..8e106163aaed 100644 --- a/fs/fuse/inode.c +++ b/fs/fuse/inode.c | |||
@@ -109,6 +109,7 @@ static int fuse_remount_fs(struct super_block *sb, int *flags, char *data) | |||
109 | 109 | ||
110 | void fuse_change_attributes(struct inode *inode, struct fuse_attr *attr) | 110 | void fuse_change_attributes(struct inode *inode, struct fuse_attr *attr) |
111 | { | 111 | { |
112 | struct fuse_conn *fc = get_fuse_conn(inode); | ||
112 | if (S_ISREG(inode->i_mode) && i_size_read(inode) != attr->size) | 113 | if (S_ISREG(inode->i_mode) && i_size_read(inode) != attr->size) |
113 | invalidate_inode_pages(inode->i_mapping); | 114 | invalidate_inode_pages(inode->i_mapping); |
114 | 115 | ||
@@ -117,7 +118,9 @@ void fuse_change_attributes(struct inode *inode, struct fuse_attr *attr) | |||
117 | inode->i_nlink = attr->nlink; | 118 | inode->i_nlink = attr->nlink; |
118 | inode->i_uid = attr->uid; | 119 | inode->i_uid = attr->uid; |
119 | inode->i_gid = attr->gid; | 120 | inode->i_gid = attr->gid; |
121 | spin_lock(&fc->lock); | ||
120 | i_size_write(inode, attr->size); | 122 | i_size_write(inode, attr->size); |
123 | spin_unlock(&fc->lock); | ||
121 | inode->i_blocks = attr->blocks; | 124 | inode->i_blocks = attr->blocks; |
122 | inode->i_atime.tv_sec = attr->atime; | 125 | inode->i_atime.tv_sec = attr->atime; |
123 | inode->i_atime.tv_nsec = attr->atimensec; | 126 | inode->i_atime.tv_nsec = attr->atimensec; |
@@ -130,7 +133,7 @@ void fuse_change_attributes(struct inode *inode, struct fuse_attr *attr) | |||
130 | static void fuse_init_inode(struct inode *inode, struct fuse_attr *attr) | 133 | static void fuse_init_inode(struct inode *inode, struct fuse_attr *attr) |
131 | { | 134 | { |
132 | inode->i_mode = attr->mode & S_IFMT; | 135 | inode->i_mode = attr->mode & S_IFMT; |
133 | i_size_write(inode, attr->size); | 136 | inode->i_size = attr->size; |
134 | if (S_ISREG(inode->i_mode)) { | 137 | if (S_ISREG(inode->i_mode)) { |
135 | fuse_init_common(inode); | 138 | fuse_init_common(inode); |
136 | fuse_init_file_inode(inode); | 139 | fuse_init_file_inode(inode); |