diff options
| author | Linus Torvalds <torvalds@linux-foundation.org> | 2012-01-09 20:37:37 -0500 |
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2012-01-09 20:37:37 -0500 |
| commit | e4e11180dfa545233e5145919b75b7fac88638df (patch) | |
| tree | bfdb18eee49aa55fd3d6170a422164e772736a1b /fs | |
| parent | 37cfc3f67db9f2d907f6bfcfae590cdbbef623e8 (diff) | |
| parent | adc0e91ab142abe93f5b0d7980ada8a7676231fe (diff) | |
Merge branch 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/viro/vfs
* 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/viro/vfs:
vfs: new helper - d_make_root()
dcache: use a dispose list in select_parent
ceph: d_alloc_root() may fail
ext4: fix failure exits
isofs: inode leak on mount failure
Diffstat (limited to 'fs')
| -rw-r--r-- | fs/ceph/super.c | 15 | ||||
| -rw-r--r-- | fs/dcache.c | 80 | ||||
| -rw-r--r-- | fs/ext4/super.c | 13 | ||||
| -rw-r--r-- | fs/isofs/inode.c | 7 |
4 files changed, 62 insertions, 53 deletions
diff --git a/fs/ceph/super.c b/fs/ceph/super.c index 11bd0fc4853f..48f61a12af66 100644 --- a/fs/ceph/super.c +++ b/fs/ceph/super.c | |||
| @@ -636,19 +636,26 @@ static struct dentry *open_root_dentry(struct ceph_fs_client *fsc, | |||
| 636 | req->r_num_caps = 2; | 636 | req->r_num_caps = 2; |
| 637 | err = ceph_mdsc_do_request(mdsc, NULL, req); | 637 | err = ceph_mdsc_do_request(mdsc, NULL, req); |
| 638 | if (err == 0) { | 638 | if (err == 0) { |
| 639 | struct inode *inode = req->r_target_inode; | ||
| 640 | req->r_target_inode = NULL; | ||
| 639 | dout("open_root_inode success\n"); | 641 | dout("open_root_inode success\n"); |
| 640 | if (ceph_ino(req->r_target_inode) == CEPH_INO_ROOT && | 642 | if (ceph_ino(inode) == CEPH_INO_ROOT && |
| 641 | fsc->sb->s_root == NULL) { | 643 | fsc->sb->s_root == NULL) { |
| 642 | root = d_alloc_root(req->r_target_inode); | 644 | root = d_alloc_root(inode); |
| 645 | if (!root) { | ||
| 646 | iput(inode); | ||
| 647 | root = ERR_PTR(-ENOMEM); | ||
| 648 | goto out; | ||
| 649 | } | ||
| 643 | ceph_init_dentry(root); | 650 | ceph_init_dentry(root); |
| 644 | } else { | 651 | } else { |
| 645 | root = d_obtain_alias(req->r_target_inode); | 652 | root = d_obtain_alias(inode); |
| 646 | } | 653 | } |
| 647 | req->r_target_inode = NULL; | ||
| 648 | dout("open_root_inode success, root dentry is %p\n", root); | 654 | dout("open_root_inode success, root dentry is %p\n", root); |
| 649 | } else { | 655 | } else { |
| 650 | root = ERR_PTR(err); | 656 | root = ERR_PTR(err); |
| 651 | } | 657 | } |
| 658 | out: | ||
| 652 | ceph_mdsc_put_request(req); | 659 | ceph_mdsc_put_request(req); |
| 653 | return root; | 660 | return root; |
| 654 | } | 661 | } |
diff --git a/fs/dcache.c b/fs/dcache.c index 9791b1e7eee4..3c6d3113a255 100644 --- a/fs/dcache.c +++ b/fs/dcache.c | |||
| @@ -276,15 +276,15 @@ static void dentry_lru_prune(struct dentry *dentry) | |||
| 276 | } | 276 | } |
| 277 | } | 277 | } |
| 278 | 278 | ||
| 279 | static void dentry_lru_move_tail(struct dentry *dentry) | 279 | static void dentry_lru_move_list(struct dentry *dentry, struct list_head *list) |
| 280 | { | 280 | { |
| 281 | spin_lock(&dcache_lru_lock); | 281 | spin_lock(&dcache_lru_lock); |
| 282 | if (list_empty(&dentry->d_lru)) { | 282 | if (list_empty(&dentry->d_lru)) { |
| 283 | list_add_tail(&dentry->d_lru, &dentry->d_sb->s_dentry_lru); | 283 | list_add_tail(&dentry->d_lru, list); |
| 284 | dentry->d_sb->s_nr_dentry_unused++; | 284 | dentry->d_sb->s_nr_dentry_unused++; |
| 285 | dentry_stat.nr_unused++; | 285 | dentry_stat.nr_unused++; |
| 286 | } else { | 286 | } else { |
| 287 | list_move_tail(&dentry->d_lru, &dentry->d_sb->s_dentry_lru); | 287 | list_move_tail(&dentry->d_lru, list); |
| 288 | } | 288 | } |
| 289 | spin_unlock(&dcache_lru_lock); | 289 | spin_unlock(&dcache_lru_lock); |
| 290 | } | 290 | } |
| @@ -770,14 +770,18 @@ static void shrink_dentry_list(struct list_head *list) | |||
| 770 | } | 770 | } |
| 771 | 771 | ||
| 772 | /** | 772 | /** |
| 773 | * __shrink_dcache_sb - shrink the dentry LRU on a given superblock | 773 | * prune_dcache_sb - shrink the dcache |
| 774 | * @sb: superblock to shrink dentry LRU. | 774 | * @sb: superblock |
| 775 | * @count: number of entries to prune | 775 | * @count: number of entries to try to free |
| 776 | * @flags: flags to control the dentry processing | 776 | * |
| 777 | * Attempt to shrink the superblock dcache LRU by @count entries. This is | ||
| 778 | * done when we need more memory an called from the superblock shrinker | ||
| 779 | * function. | ||
| 777 | * | 780 | * |
| 778 | * If flags contains DCACHE_REFERENCED reference dentries will not be pruned. | 781 | * This function may fail to free any resources if all the dentries are in |
| 782 | * use. | ||
| 779 | */ | 783 | */ |
| 780 | static void __shrink_dcache_sb(struct super_block *sb, int count, int flags) | 784 | void prune_dcache_sb(struct super_block *sb, int count) |
| 781 | { | 785 | { |
| 782 | struct dentry *dentry; | 786 | struct dentry *dentry; |
| 783 | LIST_HEAD(referenced); | 787 | LIST_HEAD(referenced); |
| @@ -796,13 +800,7 @@ relock: | |||
| 796 | goto relock; | 800 | goto relock; |
| 797 | } | 801 | } |
| 798 | 802 | ||
| 799 | /* | 803 | if (dentry->d_flags & DCACHE_REFERENCED) { |
| 800 | * If we are honouring the DCACHE_REFERENCED flag and the | ||
| 801 | * dentry has this flag set, don't free it. Clear the flag | ||
| 802 | * and put it back on the LRU. | ||
| 803 | */ | ||
| 804 | if (flags & DCACHE_REFERENCED && | ||
| 805 | dentry->d_flags & DCACHE_REFERENCED) { | ||
| 806 | dentry->d_flags &= ~DCACHE_REFERENCED; | 804 | dentry->d_flags &= ~DCACHE_REFERENCED; |
| 807 | list_move(&dentry->d_lru, &referenced); | 805 | list_move(&dentry->d_lru, &referenced); |
| 808 | spin_unlock(&dentry->d_lock); | 806 | spin_unlock(&dentry->d_lock); |
| @@ -822,23 +820,6 @@ relock: | |||
| 822 | } | 820 | } |
| 823 | 821 | ||
| 824 | /** | 822 | /** |
| 825 | * prune_dcache_sb - shrink the dcache | ||
| 826 | * @sb: superblock | ||
| 827 | * @nr_to_scan: number of entries to try to free | ||
| 828 | * | ||
| 829 | * Attempt to shrink the superblock dcache LRU by @nr_to_scan entries. This is | ||
| 830 | * done when we need more memory an called from the superblock shrinker | ||
| 831 | * function. | ||
| 832 | * | ||
| 833 | * This function may fail to free any resources if all the dentries are in | ||
| 834 | * use. | ||
| 835 | */ | ||
| 836 | void prune_dcache_sb(struct super_block *sb, int nr_to_scan) | ||
| 837 | { | ||
| 838 | __shrink_dcache_sb(sb, nr_to_scan, DCACHE_REFERENCED); | ||
| 839 | } | ||
| 840 | |||
| 841 | /** | ||
| 842 | * shrink_dcache_sb - shrink dcache for a superblock | 823 | * shrink_dcache_sb - shrink dcache for a superblock |
| 843 | * @sb: superblock | 824 | * @sb: superblock |
| 844 | * | 825 | * |
| @@ -1092,7 +1073,7 @@ EXPORT_SYMBOL(have_submounts); | |||
| 1092 | * drop the lock and return early due to latency | 1073 | * drop the lock and return early due to latency |
| 1093 | * constraints. | 1074 | * constraints. |
| 1094 | */ | 1075 | */ |
| 1095 | static int select_parent(struct dentry * parent) | 1076 | static int select_parent(struct dentry *parent, struct list_head *dispose) |
| 1096 | { | 1077 | { |
| 1097 | struct dentry *this_parent; | 1078 | struct dentry *this_parent; |
| 1098 | struct list_head *next; | 1079 | struct list_head *next; |
| @@ -1114,12 +1095,11 @@ resume: | |||
| 1114 | 1095 | ||
| 1115 | spin_lock_nested(&dentry->d_lock, DENTRY_D_LOCK_NESTED); | 1096 | spin_lock_nested(&dentry->d_lock, DENTRY_D_LOCK_NESTED); |
| 1116 | 1097 | ||
| 1117 | /* | 1098 | /* |
| 1118 | * move only zero ref count dentries to the end | 1099 | * move only zero ref count dentries to the dispose list. |
| 1119 | * of the unused list for prune_dcache | ||
| 1120 | */ | 1100 | */ |
| 1121 | if (!dentry->d_count) { | 1101 | if (!dentry->d_count) { |
| 1122 | dentry_lru_move_tail(dentry); | 1102 | dentry_lru_move_list(dentry, dispose); |
| 1123 | found++; | 1103 | found++; |
| 1124 | } else { | 1104 | } else { |
| 1125 | dentry_lru_del(dentry); | 1105 | dentry_lru_del(dentry); |
| @@ -1181,14 +1161,13 @@ rename_retry: | |||
| 1181 | * | 1161 | * |
| 1182 | * Prune the dcache to remove unused children of the parent dentry. | 1162 | * Prune the dcache to remove unused children of the parent dentry. |
| 1183 | */ | 1163 | */ |
| 1184 | |||
| 1185 | void shrink_dcache_parent(struct dentry * parent) | 1164 | void shrink_dcache_parent(struct dentry * parent) |
| 1186 | { | 1165 | { |
| 1187 | struct super_block *sb = parent->d_sb; | 1166 | LIST_HEAD(dispose); |
| 1188 | int found; | 1167 | int found; |
| 1189 | 1168 | ||
| 1190 | while ((found = select_parent(parent)) != 0) | 1169 | while ((found = select_parent(parent, &dispose)) != 0) |
| 1191 | __shrink_dcache_sb(sb, found, 0); | 1170 | shrink_dentry_list(&dispose); |
| 1192 | } | 1171 | } |
| 1193 | EXPORT_SYMBOL(shrink_dcache_parent); | 1172 | EXPORT_SYMBOL(shrink_dcache_parent); |
| 1194 | 1173 | ||
| @@ -1461,6 +1440,23 @@ struct dentry * d_alloc_root(struct inode * root_inode) | |||
| 1461 | } | 1440 | } |
| 1462 | EXPORT_SYMBOL(d_alloc_root); | 1441 | EXPORT_SYMBOL(d_alloc_root); |
| 1463 | 1442 | ||
| 1443 | struct dentry *d_make_root(struct inode *root_inode) | ||
| 1444 | { | ||
| 1445 | struct dentry *res = NULL; | ||
| 1446 | |||
| 1447 | if (root_inode) { | ||
| 1448 | static const struct qstr name = { .name = "/", .len = 1 }; | ||
| 1449 | |||
| 1450 | res = __d_alloc(root_inode->i_sb, &name); | ||
| 1451 | if (res) | ||
| 1452 | d_instantiate(res, root_inode); | ||
| 1453 | else | ||
| 1454 | iput(root_inode); | ||
| 1455 | } | ||
| 1456 | return res; | ||
| 1457 | } | ||
| 1458 | EXPORT_SYMBOL(d_make_root); | ||
| 1459 | |||
| 1464 | static struct dentry * __d_find_any_alias(struct inode *inode) | 1460 | static struct dentry * __d_find_any_alias(struct inode *inode) |
| 1465 | { | 1461 | { |
| 1466 | struct dentry *alias; | 1462 | struct dentry *alias; |
diff --git a/fs/ext4/super.c b/fs/ext4/super.c index 64e2529ae9bb..ed3ce82e2de4 100644 --- a/fs/ext4/super.c +++ b/fs/ext4/super.c | |||
| @@ -3733,10 +3733,12 @@ no_journal: | |||
| 3733 | } | 3733 | } |
| 3734 | if (!S_ISDIR(root->i_mode) || !root->i_blocks || !root->i_size) { | 3734 | if (!S_ISDIR(root->i_mode) || !root->i_blocks || !root->i_size) { |
| 3735 | ext4_msg(sb, KERN_ERR, "corrupt root inode, run e2fsck"); | 3735 | ext4_msg(sb, KERN_ERR, "corrupt root inode, run e2fsck"); |
| 3736 | iput(root); | ||
| 3736 | goto failed_mount4; | 3737 | goto failed_mount4; |
| 3737 | } | 3738 | } |
| 3738 | sb->s_root = d_alloc_root(root); | 3739 | sb->s_root = d_alloc_root(root); |
| 3739 | if (!sb->s_root) { | 3740 | if (!sb->s_root) { |
| 3741 | iput(root); | ||
| 3740 | ext4_msg(sb, KERN_ERR, "get root dentry failed"); | 3742 | ext4_msg(sb, KERN_ERR, "get root dentry failed"); |
| 3741 | ret = -ENOMEM; | 3743 | ret = -ENOMEM; |
| 3742 | goto failed_mount4; | 3744 | goto failed_mount4; |
| @@ -3773,7 +3775,7 @@ no_journal: | |||
| 3773 | if (err) { | 3775 | if (err) { |
| 3774 | ext4_msg(sb, KERN_ERR, "failed to initialize system " | 3776 | ext4_msg(sb, KERN_ERR, "failed to initialize system " |
| 3775 | "zone (%d)", err); | 3777 | "zone (%d)", err); |
| 3776 | goto failed_mount4; | 3778 | goto failed_mount4a; |
| 3777 | } | 3779 | } |
| 3778 | 3780 | ||
| 3779 | ext4_ext_init(sb); | 3781 | ext4_ext_init(sb); |
| @@ -3830,13 +3832,14 @@ cantfind_ext4: | |||
| 3830 | failed_mount7: | 3832 | failed_mount7: |
| 3831 | ext4_unregister_li_request(sb); | 3833 | ext4_unregister_li_request(sb); |
| 3832 | failed_mount6: | 3834 | failed_mount6: |
| 3833 | ext4_ext_release(sb); | ||
| 3834 | failed_mount5: | ||
| 3835 | ext4_mb_release(sb); | 3835 | ext4_mb_release(sb); |
| 3836 | failed_mount5: | ||
| 3837 | ext4_ext_release(sb); | ||
| 3836 | ext4_release_system_zone(sb); | 3838 | ext4_release_system_zone(sb); |
| 3837 | failed_mount4: | 3839 | failed_mount4a: |
| 3838 | iput(root); | 3840 | dput(sb->s_root); |
| 3839 | sb->s_root = NULL; | 3841 | sb->s_root = NULL; |
| 3842 | failed_mount4: | ||
| 3840 | ext4_msg(sb, KERN_ERR, "mount failed"); | 3843 | ext4_msg(sb, KERN_ERR, "mount failed"); |
| 3841 | destroy_workqueue(EXT4_SB(sb)->dio_unwritten_wq); | 3844 | destroy_workqueue(EXT4_SB(sb)->dio_unwritten_wq); |
| 3842 | failed_mount_wq: | 3845 | failed_mount_wq: |
diff --git a/fs/isofs/inode.c b/fs/isofs/inode.c index 7b99f5f460be..bd62c76fb5df 100644 --- a/fs/isofs/inode.c +++ b/fs/isofs/inode.c | |||
| @@ -948,8 +948,11 @@ root_found: | |||
| 948 | 948 | ||
| 949 | /* get the root dentry */ | 949 | /* get the root dentry */ |
| 950 | s->s_root = d_alloc_root(inode); | 950 | s->s_root = d_alloc_root(inode); |
| 951 | if (!(s->s_root)) | 951 | if (!(s->s_root)) { |
| 952 | goto out_no_root; | 952 | iput(inode); |
| 953 | error = -ENOMEM; | ||
| 954 | goto out_no_inode; | ||
| 955 | } | ||
| 953 | 956 | ||
| 954 | kfree(opt.iocharset); | 957 | kfree(opt.iocharset); |
| 955 | 958 | ||
