diff options
Diffstat (limited to 'fs/logfs')
-rw-r--r-- | fs/logfs/compr.c | 2 | ||||
-rw-r--r-- | fs/logfs/dev_bdev.c | 23 | ||||
-rw-r--r-- | fs/logfs/dev_mtd.c | 20 | ||||
-rw-r--r-- | fs/logfs/dir.c | 11 | ||||
-rw-r--r-- | fs/logfs/file.c | 2 | ||||
-rw-r--r-- | fs/logfs/inode.c | 11 | ||||
-rw-r--r-- | fs/logfs/journal.c | 2 | ||||
-rw-r--r-- | fs/logfs/logfs.h | 22 | ||||
-rw-r--r-- | fs/logfs/readwrite.c | 7 | ||||
-rw-r--r-- | fs/logfs/super.c | 85 |
10 files changed, 97 insertions, 88 deletions
diff --git a/fs/logfs/compr.c b/fs/logfs/compr.c index 44bbfd249abc..961f02b86d97 100644 --- a/fs/logfs/compr.c +++ b/fs/logfs/compr.c | |||
@@ -81,7 +81,7 @@ error: | |||
81 | 81 | ||
82 | int __init logfs_compr_init(void) | 82 | int __init logfs_compr_init(void) |
83 | { | 83 | { |
84 | size_t size = max(zlib_deflate_workspacesize(), | 84 | size_t size = max(zlib_deflate_workspacesize(MAX_WBITS, MAX_MEM_LEVEL), |
85 | zlib_inflate_workspacesize()); | 85 | zlib_inflate_workspacesize()); |
86 | stream.workspace = vmalloc(size); | 86 | stream.workspace = vmalloc(size); |
87 | if (!stream.workspace) | 87 | if (!stream.workspace) |
diff --git a/fs/logfs/dev_bdev.c b/fs/logfs/dev_bdev.c index 9bd2ce2a3040..df0de27c2733 100644 --- a/fs/logfs/dev_bdev.c +++ b/fs/logfs/dev_bdev.c | |||
@@ -10,6 +10,7 @@ | |||
10 | #include <linux/blkdev.h> | 10 | #include <linux/blkdev.h> |
11 | #include <linux/buffer_head.h> | 11 | #include <linux/buffer_head.h> |
12 | #include <linux/gfp.h> | 12 | #include <linux/gfp.h> |
13 | #include <linux/prefetch.h> | ||
13 | 14 | ||
14 | #define PAGE_OFS(ofs) ((ofs) & (PAGE_SIZE-1)) | 15 | #define PAGE_OFS(ofs) ((ofs) & (PAGE_SIZE-1)) |
15 | 16 | ||
@@ -39,7 +40,6 @@ static int sync_request(struct page *page, struct block_device *bdev, int rw) | |||
39 | bio.bi_end_io = request_complete; | 40 | bio.bi_end_io = request_complete; |
40 | 41 | ||
41 | submit_bio(rw, &bio); | 42 | submit_bio(rw, &bio); |
42 | generic_unplug_device(bdev_get_queue(bdev)); | ||
43 | wait_for_completion(&complete); | 43 | wait_for_completion(&complete); |
44 | return test_bit(BIO_UPTODATE, &bio.bi_flags) ? 0 : -EIO; | 44 | return test_bit(BIO_UPTODATE, &bio.bi_flags) ? 0 : -EIO; |
45 | } | 45 | } |
@@ -168,7 +168,6 @@ static void bdev_writeseg(struct super_block *sb, u64 ofs, size_t len) | |||
168 | } | 168 | } |
169 | len = PAGE_ALIGN(len); | 169 | len = PAGE_ALIGN(len); |
170 | __bdev_writeseg(sb, ofs, ofs >> PAGE_SHIFT, len >> PAGE_SHIFT); | 170 | __bdev_writeseg(sb, ofs, ofs >> PAGE_SHIFT, len >> PAGE_SHIFT); |
171 | generic_unplug_device(bdev_get_queue(logfs_super(sb)->s_bdev)); | ||
172 | } | 171 | } |
173 | 172 | ||
174 | 173 | ||
@@ -298,9 +297,9 @@ static int bdev_write_sb(struct super_block *sb, struct page *page) | |||
298 | return sync_request(page, bdev, WRITE); | 297 | return sync_request(page, bdev, WRITE); |
299 | } | 298 | } |
300 | 299 | ||
301 | static void bdev_put_device(struct super_block *sb) | 300 | static void bdev_put_device(struct logfs_super *s) |
302 | { | 301 | { |
303 | close_bdev_exclusive(logfs_super(sb)->s_bdev, FMODE_READ|FMODE_WRITE); | 302 | blkdev_put(s->s_bdev, FMODE_READ|FMODE_WRITE|FMODE_EXCL); |
304 | } | 303 | } |
305 | 304 | ||
306 | static int bdev_can_write_buf(struct super_block *sb, u64 ofs) | 305 | static int bdev_can_write_buf(struct super_block *sb, u64 ofs) |
@@ -320,20 +319,24 @@ static const struct logfs_device_ops bd_devops = { | |||
320 | .put_device = bdev_put_device, | 319 | .put_device = bdev_put_device, |
321 | }; | 320 | }; |
322 | 321 | ||
323 | int logfs_get_sb_bdev(struct file_system_type *type, int flags, | 322 | int logfs_get_sb_bdev(struct logfs_super *p, struct file_system_type *type, |
324 | const char *devname, struct vfsmount *mnt) | 323 | const char *devname) |
325 | { | 324 | { |
326 | struct block_device *bdev; | 325 | struct block_device *bdev; |
327 | 326 | ||
328 | bdev = open_bdev_exclusive(devname, FMODE_READ|FMODE_WRITE, type); | 327 | bdev = blkdev_get_by_path(devname, FMODE_READ|FMODE_WRITE|FMODE_EXCL, |
328 | type); | ||
329 | if (IS_ERR(bdev)) | 329 | if (IS_ERR(bdev)) |
330 | return PTR_ERR(bdev); | 330 | return PTR_ERR(bdev); |
331 | 331 | ||
332 | if (MAJOR(bdev->bd_dev) == MTD_BLOCK_MAJOR) { | 332 | if (MAJOR(bdev->bd_dev) == MTD_BLOCK_MAJOR) { |
333 | int mtdnr = MINOR(bdev->bd_dev); | 333 | int mtdnr = MINOR(bdev->bd_dev); |
334 | close_bdev_exclusive(bdev, FMODE_READ|FMODE_WRITE); | 334 | blkdev_put(bdev, FMODE_READ|FMODE_WRITE|FMODE_EXCL); |
335 | return logfs_get_sb_mtd(type, flags, mtdnr, mnt); | 335 | return logfs_get_sb_mtd(p, mtdnr); |
336 | } | 336 | } |
337 | 337 | ||
338 | return logfs_get_sb_device(type, flags, NULL, bdev, &bd_devops, mnt); | 338 | p->s_bdev = bdev; |
339 | p->s_mtd = NULL; | ||
340 | p->s_devops = &bd_devops; | ||
341 | return 0; | ||
339 | } | 342 | } |
diff --git a/fs/logfs/dev_mtd.c b/fs/logfs/dev_mtd.c index a85d47d13e4b..339e17e9133d 100644 --- a/fs/logfs/dev_mtd.c +++ b/fs/logfs/dev_mtd.c | |||
@@ -60,7 +60,7 @@ static int mtd_write(struct super_block *sb, loff_t ofs, size_t len, void *buf) | |||
60 | * asynchronous properties. So just to prevent the first implementor of such | 60 | * asynchronous properties. So just to prevent the first implementor of such |
61 | * a thing from breaking logfs in 2350, we do the usual pointless dance to | 61 | * a thing from breaking logfs in 2350, we do the usual pointless dance to |
62 | * declare a completion variable and wait for completion before returning | 62 | * declare a completion variable and wait for completion before returning |
63 | * from mtd_erase(). What an excercise in futility! | 63 | * from mtd_erase(). What an exercise in futility! |
64 | */ | 64 | */ |
65 | static void logfs_erase_callback(struct erase_info *ei) | 65 | static void logfs_erase_callback(struct erase_info *ei) |
66 | { | 66 | { |
@@ -230,9 +230,9 @@ static void mtd_writeseg(struct super_block *sb, u64 ofs, size_t len) | |||
230 | __mtd_writeseg(sb, ofs, ofs >> PAGE_SHIFT, len >> PAGE_SHIFT); | 230 | __mtd_writeseg(sb, ofs, ofs >> PAGE_SHIFT, len >> PAGE_SHIFT); |
231 | } | 231 | } |
232 | 232 | ||
233 | static void mtd_put_device(struct super_block *sb) | 233 | static void mtd_put_device(struct logfs_super *s) |
234 | { | 234 | { |
235 | put_mtd_device(logfs_super(sb)->s_mtd); | 235 | put_mtd_device(s->s_mtd); |
236 | } | 236 | } |
237 | 237 | ||
238 | static int mtd_can_write_buf(struct super_block *sb, u64 ofs) | 238 | static int mtd_can_write_buf(struct super_block *sb, u64 ofs) |
@@ -265,14 +265,14 @@ static const struct logfs_device_ops mtd_devops = { | |||
265 | .put_device = mtd_put_device, | 265 | .put_device = mtd_put_device, |
266 | }; | 266 | }; |
267 | 267 | ||
268 | int logfs_get_sb_mtd(struct file_system_type *type, int flags, | 268 | int logfs_get_sb_mtd(struct logfs_super *s, int mtdnr) |
269 | int mtdnr, struct vfsmount *mnt) | ||
270 | { | 269 | { |
271 | struct mtd_info *mtd; | 270 | struct mtd_info *mtd = get_mtd_device(NULL, mtdnr); |
272 | const struct logfs_device_ops *devops = &mtd_devops; | ||
273 | |||
274 | mtd = get_mtd_device(NULL, mtdnr); | ||
275 | if (IS_ERR(mtd)) | 271 | if (IS_ERR(mtd)) |
276 | return PTR_ERR(mtd); | 272 | return PTR_ERR(mtd); |
277 | return logfs_get_sb_device(type, flags, mtd, NULL, devops, mnt); | 273 | |
274 | s->s_bdev = NULL; | ||
275 | s->s_mtd = mtd; | ||
276 | s->s_devops = &mtd_devops; | ||
277 | return 0; | ||
278 | } | 278 | } |
diff --git a/fs/logfs/dir.c b/fs/logfs/dir.c index 9777eb5b5522..1afae26cf236 100644 --- a/fs/logfs/dir.c +++ b/fs/logfs/dir.c | |||
@@ -92,7 +92,7 @@ static int beyond_eof(struct inode *inode, loff_t bix) | |||
92 | * so short names (len <= 9) don't even occupy the complete 32bit name | 92 | * so short names (len <= 9) don't even occupy the complete 32bit name |
93 | * space. A prime >256 ensures short names quickly spread the 32bit | 93 | * space. A prime >256 ensures short names quickly spread the 32bit |
94 | * name space. Add about 26 for the estimated amount of information | 94 | * name space. Add about 26 for the estimated amount of information |
95 | * of each character and pick a prime nearby, preferrably a bit-sparse | 95 | * of each character and pick a prime nearby, preferably a bit-sparse |
96 | * one. | 96 | * one. |
97 | */ | 97 | */ |
98 | static u32 hash_32(const char *s, int len, u32 seed) | 98 | static u32 hash_32(const char *s, int len, u32 seed) |
@@ -555,11 +555,6 @@ static int logfs_symlink(struct inode *dir, struct dentry *dentry, | |||
555 | return __logfs_create(dir, dentry, inode, target, destlen); | 555 | return __logfs_create(dir, dentry, inode, target, destlen); |
556 | } | 556 | } |
557 | 557 | ||
558 | static int logfs_permission(struct inode *inode, int mask) | ||
559 | { | ||
560 | return generic_permission(inode, mask, NULL); | ||
561 | } | ||
562 | |||
563 | static int logfs_link(struct dentry *old_dentry, struct inode *dir, | 558 | static int logfs_link(struct dentry *old_dentry, struct inode *dir, |
564 | struct dentry *dentry) | 559 | struct dentry *dentry) |
565 | { | 560 | { |
@@ -569,7 +564,7 @@ static int logfs_link(struct dentry *old_dentry, struct inode *dir, | |||
569 | return -EMLINK; | 564 | return -EMLINK; |
570 | 565 | ||
571 | inode->i_ctime = dir->i_ctime = dir->i_mtime = CURRENT_TIME; | 566 | inode->i_ctime = dir->i_ctime = dir->i_mtime = CURRENT_TIME; |
572 | atomic_inc(&inode->i_count); | 567 | ihold(inode); |
573 | inode->i_nlink++; | 568 | inode->i_nlink++; |
574 | mark_inode_dirty_sync(inode); | 569 | mark_inode_dirty_sync(inode); |
575 | 570 | ||
@@ -818,7 +813,6 @@ const struct inode_operations logfs_dir_iops = { | |||
818 | .mknod = logfs_mknod, | 813 | .mknod = logfs_mknod, |
819 | .rename = logfs_rename, | 814 | .rename = logfs_rename, |
820 | .rmdir = logfs_rmdir, | 815 | .rmdir = logfs_rmdir, |
821 | .permission = logfs_permission, | ||
822 | .symlink = logfs_symlink, | 816 | .symlink = logfs_symlink, |
823 | .unlink = logfs_unlink, | 817 | .unlink = logfs_unlink, |
824 | }; | 818 | }; |
@@ -827,4 +821,5 @@ const struct file_operations logfs_dir_fops = { | |||
827 | .unlocked_ioctl = logfs_ioctl, | 821 | .unlocked_ioctl = logfs_ioctl, |
828 | .readdir = logfs_readdir, | 822 | .readdir = logfs_readdir, |
829 | .read = generic_read_dir, | 823 | .read = generic_read_dir, |
824 | .llseek = default_llseek, | ||
830 | }; | 825 | }; |
diff --git a/fs/logfs/file.c b/fs/logfs/file.c index e86376b87af1..c2ad7028def4 100644 --- a/fs/logfs/file.c +++ b/fs/logfs/file.c | |||
@@ -196,7 +196,7 @@ long logfs_ioctl(struct file *file, unsigned int cmd, unsigned long arg) | |||
196 | if (IS_RDONLY(inode)) | 196 | if (IS_RDONLY(inode)) |
197 | return -EROFS; | 197 | return -EROFS; |
198 | 198 | ||
199 | if (!is_owner_or_cap(inode)) | 199 | if (!inode_owner_or_capable(inode)) |
200 | return -EACCES; | 200 | return -EACCES; |
201 | 201 | ||
202 | err = get_user(flags, (int __user *)arg); | 202 | err = get_user(flags, (int __user *)arg); |
diff --git a/fs/logfs/inode.c b/fs/logfs/inode.c index d8c71ece098f..edfea7a3a747 100644 --- a/fs/logfs/inode.c +++ b/fs/logfs/inode.c | |||
@@ -141,13 +141,20 @@ struct inode *logfs_safe_iget(struct super_block *sb, ino_t ino, int *is_cached) | |||
141 | return __logfs_iget(sb, ino); | 141 | return __logfs_iget(sb, ino); |
142 | } | 142 | } |
143 | 143 | ||
144 | static void logfs_i_callback(struct rcu_head *head) | ||
145 | { | ||
146 | struct inode *inode = container_of(head, struct inode, i_rcu); | ||
147 | INIT_LIST_HEAD(&inode->i_dentry); | ||
148 | kmem_cache_free(logfs_inode_cache, logfs_inode(inode)); | ||
149 | } | ||
150 | |||
144 | static void __logfs_destroy_inode(struct inode *inode) | 151 | static void __logfs_destroy_inode(struct inode *inode) |
145 | { | 152 | { |
146 | struct logfs_inode *li = logfs_inode(inode); | 153 | struct logfs_inode *li = logfs_inode(inode); |
147 | 154 | ||
148 | BUG_ON(li->li_block); | 155 | BUG_ON(li->li_block); |
149 | list_del(&li->li_freeing_list); | 156 | list_del(&li->li_freeing_list); |
150 | kmem_cache_free(logfs_inode_cache, li); | 157 | call_rcu(&inode->i_rcu, logfs_i_callback); |
151 | } | 158 | } |
152 | 159 | ||
153 | static void logfs_destroy_inode(struct inode *inode) | 160 | static void logfs_destroy_inode(struct inode *inode) |
@@ -286,7 +293,7 @@ static int logfs_write_inode(struct inode *inode, struct writeback_control *wbc) | |||
286 | return ret; | 293 | return ret; |
287 | } | 294 | } |
288 | 295 | ||
289 | /* called with inode_lock held */ | 296 | /* called with inode->i_lock held */ |
290 | static int logfs_drop_inode(struct inode *inode) | 297 | static int logfs_drop_inode(struct inode *inode) |
291 | { | 298 | { |
292 | struct logfs_super *super = logfs_super(inode->i_sb); | 299 | struct logfs_super *super = logfs_super(inode->i_sb); |
diff --git a/fs/logfs/journal.c b/fs/logfs/journal.c index f46ee8b0e135..9da29706f91c 100644 --- a/fs/logfs/journal.c +++ b/fs/logfs/journal.c | |||
@@ -828,7 +828,7 @@ void do_logfs_journal_wl_pass(struct super_block *sb) | |||
828 | super->s_journal_seg[i] = segno; | 828 | super->s_journal_seg[i] = segno; |
829 | super->s_journal_ec[i] = ec; | 829 | super->s_journal_ec[i] = ec; |
830 | logfs_set_segment_reserved(sb, segno); | 830 | logfs_set_segment_reserved(sb, segno); |
831 | err = btree_insert32(head, segno, (void *)1, GFP_KERNEL); | 831 | err = btree_insert32(head, segno, (void *)1, GFP_NOFS); |
832 | BUG_ON(err); /* mempool should prevent this */ | 832 | BUG_ON(err); /* mempool should prevent this */ |
833 | err = logfs_erase_segment(sb, segno, 1); | 833 | err = logfs_erase_segment(sb, segno, 1); |
834 | BUG_ON(err); /* FIXME: remount-ro would be nicer */ | 834 | BUG_ON(err); /* FIXME: remount-ro would be nicer */ |
diff --git a/fs/logfs/logfs.h b/fs/logfs/logfs.h index b8786264d243..57afd4a6fabb 100644 --- a/fs/logfs/logfs.h +++ b/fs/logfs/logfs.h | |||
@@ -136,6 +136,7 @@ struct logfs_area_ops { | |||
136 | int (*erase_segment)(struct logfs_area *area); | 136 | int (*erase_segment)(struct logfs_area *area); |
137 | }; | 137 | }; |
138 | 138 | ||
139 | struct logfs_super; /* forward */ | ||
139 | /** | 140 | /** |
140 | * struct logfs_device_ops - device access operations | 141 | * struct logfs_device_ops - device access operations |
141 | * | 142 | * |
@@ -156,7 +157,7 @@ struct logfs_device_ops { | |||
156 | int ensure_write); | 157 | int ensure_write); |
157 | int (*can_write_buf)(struct super_block *sb, u64 ofs); | 158 | int (*can_write_buf)(struct super_block *sb, u64 ofs); |
158 | void (*sync)(struct super_block *sb); | 159 | void (*sync)(struct super_block *sb); |
159 | void (*put_device)(struct super_block *sb); | 160 | void (*put_device)(struct logfs_super *s); |
160 | }; | 161 | }; |
161 | 162 | ||
162 | /** | 163 | /** |
@@ -471,11 +472,13 @@ void logfs_compr_exit(void); | |||
471 | 472 | ||
472 | /* dev_bdev.c */ | 473 | /* dev_bdev.c */ |
473 | #ifdef CONFIG_BLOCK | 474 | #ifdef CONFIG_BLOCK |
474 | int logfs_get_sb_bdev(struct file_system_type *type, int flags, | 475 | int logfs_get_sb_bdev(struct logfs_super *s, |
475 | const char *devname, struct vfsmount *mnt); | 476 | struct file_system_type *type, |
477 | const char *devname); | ||
476 | #else | 478 | #else |
477 | static inline int logfs_get_sb_bdev(struct file_system_type *type, int flags, | 479 | static inline int logfs_get_sb_bdev(struct logfs_super *s, |
478 | const char *devname, struct vfsmount *mnt) | 480 | struct file_system_type *type, |
481 | const char *devname) | ||
479 | { | 482 | { |
480 | return -ENODEV; | 483 | return -ENODEV; |
481 | } | 484 | } |
@@ -483,11 +486,9 @@ static inline int logfs_get_sb_bdev(struct file_system_type *type, int flags, | |||
483 | 486 | ||
484 | /* dev_mtd.c */ | 487 | /* dev_mtd.c */ |
485 | #ifdef CONFIG_MTD | 488 | #ifdef CONFIG_MTD |
486 | int logfs_get_sb_mtd(struct file_system_type *type, int flags, | 489 | int logfs_get_sb_mtd(struct logfs_super *s, int mtdnr); |
487 | int mtdnr, struct vfsmount *mnt); | ||
488 | #else | 490 | #else |
489 | static inline int logfs_get_sb_mtd(struct file_system_type *type, int flags, | 491 | static inline int logfs_get_sb_mtd(struct logfs_super *s, int mtdnr) |
490 | int mtdnr, struct vfsmount *mnt) | ||
491 | { | 492 | { |
492 | return -ENODEV; | 493 | return -ENODEV; |
493 | } | 494 | } |
@@ -619,9 +620,6 @@ void emergency_read_end(struct page *page); | |||
619 | void logfs_crash_dump(struct super_block *sb); | 620 | void logfs_crash_dump(struct super_block *sb); |
620 | void *memchr_inv(const void *s, int c, size_t n); | 621 | void *memchr_inv(const void *s, int c, size_t n); |
621 | int logfs_statfs(struct dentry *dentry, struct kstatfs *stats); | 622 | int logfs_statfs(struct dentry *dentry, struct kstatfs *stats); |
622 | int logfs_get_sb_device(struct file_system_type *type, int flags, | ||
623 | struct mtd_info *mtd, struct block_device *bdev, | ||
624 | const struct logfs_device_ops *devops, struct vfsmount *mnt); | ||
625 | int logfs_check_ds(struct logfs_disk_super *ds); | 623 | int logfs_check_ds(struct logfs_disk_super *ds); |
626 | int logfs_write_sb(struct super_block *sb); | 624 | int logfs_write_sb(struct super_block *sb); |
627 | 625 | ||
diff --git a/fs/logfs/readwrite.c b/fs/logfs/readwrite.c index 6127baf0e188..d8d09380c7de 100644 --- a/fs/logfs/readwrite.c +++ b/fs/logfs/readwrite.c | |||
@@ -481,7 +481,7 @@ static int inode_write_alias(struct super_block *sb, | |||
481 | val = inode_val0(inode); | 481 | val = inode_val0(inode); |
482 | break; | 482 | break; |
483 | case INODE_USED_OFS: | 483 | case INODE_USED_OFS: |
484 | val = cpu_to_be64(li->li_used_bytes);; | 484 | val = cpu_to_be64(li->li_used_bytes); |
485 | break; | 485 | break; |
486 | case INODE_SIZE_OFS: | 486 | case INODE_SIZE_OFS: |
487 | val = cpu_to_be64(i_size_read(inode)); | 487 | val = cpu_to_be64(i_size_read(inode)); |
@@ -1616,7 +1616,7 @@ int logfs_rewrite_block(struct inode *inode, u64 bix, u64 ofs, | |||
1616 | err = logfs_write_buf(inode, page, flags); | 1616 | err = logfs_write_buf(inode, page, flags); |
1617 | if (!err && shrink_level(gc_level) == 0) { | 1617 | if (!err && shrink_level(gc_level) == 0) { |
1618 | /* Rewrite cannot mark the inode dirty but has to | 1618 | /* Rewrite cannot mark the inode dirty but has to |
1619 | * write it immediatly. | 1619 | * write it immediately. |
1620 | * Q: Can't we just create an alias for the inode | 1620 | * Q: Can't we just create an alias for the inode |
1621 | * instead? And if not, why not? | 1621 | * instead? And if not, why not? |
1622 | */ | 1622 | */ |
@@ -1994,6 +1994,9 @@ static int do_write_inode(struct inode *inode) | |||
1994 | 1994 | ||
1995 | /* FIXME: transaction is part of logfs_block now. Is that enough? */ | 1995 | /* FIXME: transaction is part of logfs_block now. Is that enough? */ |
1996 | err = logfs_write_buf(master_inode, page, 0); | 1996 | err = logfs_write_buf(master_inode, page, 0); |
1997 | if (err) | ||
1998 | move_page_to_inode(inode, page); | ||
1999 | |||
1997 | logfs_put_write_page(page); | 2000 | logfs_put_write_page(page); |
1998 | return err; | 2001 | return err; |
1999 | } | 2002 | } |
diff --git a/fs/logfs/super.c b/fs/logfs/super.c index 5336155c5d81..ce03a182c771 100644 --- a/fs/logfs/super.c +++ b/fs/logfs/super.c | |||
@@ -325,7 +325,7 @@ static int logfs_make_writeable(struct super_block *sb) | |||
325 | return 0; | 325 | return 0; |
326 | } | 326 | } |
327 | 327 | ||
328 | static int logfs_get_sb_final(struct super_block *sb, struct vfsmount *mnt) | 328 | static int logfs_get_sb_final(struct super_block *sb) |
329 | { | 329 | { |
330 | struct logfs_super *super = logfs_super(sb); | 330 | struct logfs_super *super = logfs_super(sb); |
331 | struct inode *rootdir; | 331 | struct inode *rootdir; |
@@ -356,7 +356,6 @@ static int logfs_get_sb_final(struct super_block *sb, struct vfsmount *mnt) | |||
356 | } | 356 | } |
357 | 357 | ||
358 | log_super("LogFS: Finished mounting\n"); | 358 | log_super("LogFS: Finished mounting\n"); |
359 | simple_set_mnt(mnt, sb); | ||
360 | return 0; | 359 | return 0; |
361 | 360 | ||
362 | fail: | 361 | fail: |
@@ -481,10 +480,6 @@ static int logfs_read_sb(struct super_block *sb, int read_only) | |||
481 | !read_only) | 480 | !read_only) |
482 | return -EIO; | 481 | return -EIO; |
483 | 482 | ||
484 | mutex_init(&super->s_dirop_mutex); | ||
485 | mutex_init(&super->s_object_alias_mutex); | ||
486 | INIT_LIST_HEAD(&super->s_freeing_list); | ||
487 | |||
488 | ret = logfs_init_rw(sb); | 483 | ret = logfs_init_rw(sb); |
489 | if (ret) | 484 | if (ret) |
490 | return ret; | 485 | return ret; |
@@ -529,43 +524,37 @@ static void logfs_kill_sb(struct super_block *sb) | |||
529 | logfs_cleanup_rw(sb); | 524 | logfs_cleanup_rw(sb); |
530 | if (super->s_erase_page) | 525 | if (super->s_erase_page) |
531 | __free_page(super->s_erase_page); | 526 | __free_page(super->s_erase_page); |
532 | super->s_devops->put_device(sb); | 527 | super->s_devops->put_device(super); |
533 | logfs_mempool_destroy(super->s_btree_pool); | 528 | logfs_mempool_destroy(super->s_btree_pool); |
534 | logfs_mempool_destroy(super->s_alias_pool); | 529 | logfs_mempool_destroy(super->s_alias_pool); |
535 | kfree(super); | 530 | kfree(super); |
536 | log_super("LogFS: Finished unmounting\n"); | 531 | log_super("LogFS: Finished unmounting\n"); |
537 | } | 532 | } |
538 | 533 | ||
539 | int logfs_get_sb_device(struct file_system_type *type, int flags, | 534 | static struct dentry *logfs_get_sb_device(struct logfs_super *super, |
540 | struct mtd_info *mtd, struct block_device *bdev, | 535 | struct file_system_type *type, int flags) |
541 | const struct logfs_device_ops *devops, struct vfsmount *mnt) | ||
542 | { | 536 | { |
543 | struct logfs_super *super; | ||
544 | struct super_block *sb; | 537 | struct super_block *sb; |
545 | int err = -ENOMEM; | 538 | int err = -ENOMEM; |
546 | static int mount_count; | 539 | static int mount_count; |
547 | 540 | ||
548 | log_super("LogFS: Start mount %x\n", mount_count++); | 541 | log_super("LogFS: Start mount %x\n", mount_count++); |
549 | super = kzalloc(sizeof(*super), GFP_KERNEL); | ||
550 | if (!super) | ||
551 | goto err0; | ||
552 | 542 | ||
553 | super->s_mtd = mtd; | ||
554 | super->s_bdev = bdev; | ||
555 | err = -EINVAL; | 543 | err = -EINVAL; |
556 | sb = sget(type, logfs_sb_test, logfs_sb_set, super); | 544 | sb = sget(type, logfs_sb_test, logfs_sb_set, super); |
557 | if (IS_ERR(sb)) | 545 | if (IS_ERR(sb)) { |
558 | goto err0; | 546 | super->s_devops->put_device(super); |
547 | kfree(super); | ||
548 | return ERR_CAST(sb); | ||
549 | } | ||
559 | 550 | ||
560 | if (sb->s_root) { | 551 | if (sb->s_root) { |
561 | /* Device is already in use */ | 552 | /* Device is already in use */ |
562 | err = 0; | 553 | super->s_devops->put_device(super); |
563 | simple_set_mnt(mnt, sb); | 554 | kfree(super); |
564 | goto err0; | 555 | return dget(sb->s_root); |
565 | } | 556 | } |
566 | 557 | ||
567 | super->s_devops = devops; | ||
568 | |||
569 | /* | 558 | /* |
570 | * sb->s_maxbytes is limited to 8TB. On 32bit systems, the page cache | 559 | * sb->s_maxbytes is limited to 8TB. On 32bit systems, the page cache |
571 | * only covers 16TB and the upper 8TB are used for indirect blocks. | 560 | * only covers 16TB and the upper 8TB are used for indirect blocks. |
@@ -581,10 +570,12 @@ int logfs_get_sb_device(struct file_system_type *type, int flags, | |||
581 | goto err1; | 570 | goto err1; |
582 | 571 | ||
583 | sb->s_flags |= MS_ACTIVE; | 572 | sb->s_flags |= MS_ACTIVE; |
584 | err = logfs_get_sb_final(sb, mnt); | 573 | err = logfs_get_sb_final(sb); |
585 | if (err) | 574 | if (err) { |
586 | deactivate_locked_super(sb); | 575 | deactivate_locked_super(sb); |
587 | return err; | 576 | return ERR_PTR(err); |
577 | } | ||
578 | return dget(sb->s_root); | ||
588 | 579 | ||
589 | err1: | 580 | err1: |
590 | /* no ->s_root, no ->put_super() */ | 581 | /* no ->s_root, no ->put_super() */ |
@@ -592,37 +583,49 @@ err1: | |||
592 | iput(super->s_segfile_inode); | 583 | iput(super->s_segfile_inode); |
593 | iput(super->s_mapping_inode); | 584 | iput(super->s_mapping_inode); |
594 | deactivate_locked_super(sb); | 585 | deactivate_locked_super(sb); |
595 | return err; | 586 | return ERR_PTR(err); |
596 | err0: | ||
597 | kfree(super); | ||
598 | //devops->put_device(sb); | ||
599 | return err; | ||
600 | } | 587 | } |
601 | 588 | ||
602 | static int logfs_get_sb(struct file_system_type *type, int flags, | 589 | static struct dentry *logfs_mount(struct file_system_type *type, int flags, |
603 | const char *devname, void *data, struct vfsmount *mnt) | 590 | const char *devname, void *data) |
604 | { | 591 | { |
605 | ulong mtdnr; | 592 | ulong mtdnr; |
593 | struct logfs_super *super; | ||
594 | int err; | ||
606 | 595 | ||
607 | if (!devname) | 596 | super = kzalloc(sizeof(*super), GFP_KERNEL); |
608 | return logfs_get_sb_bdev(type, flags, devname, mnt); | 597 | if (!super) |
609 | if (strncmp(devname, "mtd", 3)) | 598 | return ERR_PTR(-ENOMEM); |
610 | return logfs_get_sb_bdev(type, flags, devname, mnt); | ||
611 | 599 | ||
612 | { | 600 | mutex_init(&super->s_dirop_mutex); |
601 | mutex_init(&super->s_object_alias_mutex); | ||
602 | INIT_LIST_HEAD(&super->s_freeing_list); | ||
603 | |||
604 | if (!devname) | ||
605 | err = logfs_get_sb_bdev(super, type, devname); | ||
606 | else if (strncmp(devname, "mtd", 3)) | ||
607 | err = logfs_get_sb_bdev(super, type, devname); | ||
608 | else { | ||
613 | char *garbage; | 609 | char *garbage; |
614 | mtdnr = simple_strtoul(devname+3, &garbage, 0); | 610 | mtdnr = simple_strtoul(devname+3, &garbage, 0); |
615 | if (*garbage) | 611 | if (*garbage) |
616 | return -EINVAL; | 612 | err = -EINVAL; |
613 | else | ||
614 | err = logfs_get_sb_mtd(super, mtdnr); | ||
615 | } | ||
616 | |||
617 | if (err) { | ||
618 | kfree(super); | ||
619 | return ERR_PTR(err); | ||
617 | } | 620 | } |
618 | 621 | ||
619 | return logfs_get_sb_mtd(type, flags, mtdnr, mnt); | 622 | return logfs_get_sb_device(super, type, flags); |
620 | } | 623 | } |
621 | 624 | ||
622 | static struct file_system_type logfs_fs_type = { | 625 | static struct file_system_type logfs_fs_type = { |
623 | .owner = THIS_MODULE, | 626 | .owner = THIS_MODULE, |
624 | .name = "logfs", | 627 | .name = "logfs", |
625 | .get_sb = logfs_get_sb, | 628 | .mount = logfs_mount, |
626 | .kill_sb = logfs_kill_sb, | 629 | .kill_sb = logfs_kill_sb, |
627 | .fs_flags = FS_REQUIRES_DEV, | 630 | .fs_flags = FS_REQUIRES_DEV, |
628 | 631 | ||