diff options
Diffstat (limited to 'fs/block_dev.c')
| -rw-r--r-- | fs/block_dev.c | 114 |
1 files changed, 56 insertions, 58 deletions
diff --git a/fs/block_dev.c b/fs/block_dev.c index 37534573960b..045f98854f14 100644 --- a/fs/block_dev.c +++ b/fs/block_dev.c | |||
| @@ -884,6 +884,61 @@ void bd_set_size(struct block_device *bdev, loff_t size) | |||
| 884 | } | 884 | } |
| 885 | EXPORT_SYMBOL(bd_set_size); | 885 | EXPORT_SYMBOL(bd_set_size); |
| 886 | 886 | ||
| 887 | static int __blkdev_put(struct block_device *bdev, unsigned int subclass) | ||
| 888 | { | ||
| 889 | int ret = 0; | ||
| 890 | struct inode *bd_inode = bdev->bd_inode; | ||
| 891 | struct gendisk *disk = bdev->bd_disk; | ||
| 892 | |||
| 893 | mutex_lock_nested(&bdev->bd_mutex, subclass); | ||
| 894 | lock_kernel(); | ||
| 895 | if (!--bdev->bd_openers) { | ||
| 896 | sync_blockdev(bdev); | ||
| 897 | kill_bdev(bdev); | ||
| 898 | } | ||
| 899 | if (bdev->bd_contains == bdev) { | ||
| 900 | if (disk->fops->release) | ||
| 901 | ret = disk->fops->release(bd_inode, NULL); | ||
| 902 | } else { | ||
| 903 | mutex_lock_nested(&bdev->bd_contains->bd_mutex, | ||
| 904 | subclass + 1); | ||
| 905 | bdev->bd_contains->bd_part_count--; | ||
| 906 | mutex_unlock(&bdev->bd_contains->bd_mutex); | ||
| 907 | } | ||
| 908 | if (!bdev->bd_openers) { | ||
| 909 | struct module *owner = disk->fops->owner; | ||
| 910 | |||
| 911 | put_disk(disk); | ||
| 912 | module_put(owner); | ||
| 913 | |||
| 914 | if (bdev->bd_contains != bdev) { | ||
| 915 | kobject_put(&bdev->bd_part->kobj); | ||
| 916 | bdev->bd_part = NULL; | ||
| 917 | } | ||
| 918 | bdev->bd_disk = NULL; | ||
| 919 | bdev->bd_inode->i_data.backing_dev_info = &default_backing_dev_info; | ||
| 920 | if (bdev != bdev->bd_contains) | ||
| 921 | __blkdev_put(bdev->bd_contains, subclass + 1); | ||
| 922 | bdev->bd_contains = NULL; | ||
| 923 | } | ||
| 924 | unlock_kernel(); | ||
| 925 | mutex_unlock(&bdev->bd_mutex); | ||
| 926 | bdput(bdev); | ||
| 927 | return ret; | ||
| 928 | } | ||
| 929 | |||
| 930 | int blkdev_put(struct block_device *bdev) | ||
| 931 | { | ||
| 932 | return __blkdev_put(bdev, BD_MUTEX_NORMAL); | ||
| 933 | } | ||
| 934 | EXPORT_SYMBOL(blkdev_put); | ||
| 935 | |||
| 936 | int blkdev_put_partition(struct block_device *bdev) | ||
| 937 | { | ||
| 938 | return __blkdev_put(bdev, BD_MUTEX_PARTITION); | ||
| 939 | } | ||
| 940 | EXPORT_SYMBOL(blkdev_put_partition); | ||
| 941 | |||
| 887 | static int | 942 | static int |
| 888 | blkdev_get_whole(struct block_device *bdev, mode_t mode, unsigned flags); | 943 | blkdev_get_whole(struct block_device *bdev, mode_t mode, unsigned flags); |
| 889 | 944 | ||
| @@ -980,7 +1035,7 @@ out_first: | |||
| 980 | bdev->bd_disk = NULL; | 1035 | bdev->bd_disk = NULL; |
| 981 | bdev->bd_inode->i_data.backing_dev_info = &default_backing_dev_info; | 1036 | bdev->bd_inode->i_data.backing_dev_info = &default_backing_dev_info; |
| 982 | if (bdev != bdev->bd_contains) | 1037 | if (bdev != bdev->bd_contains) |
| 983 | blkdev_put(bdev->bd_contains); | 1038 | __blkdev_put(bdev->bd_contains, BD_MUTEX_WHOLE); |
| 984 | bdev->bd_contains = NULL; | 1039 | bdev->bd_contains = NULL; |
| 985 | put_disk(disk); | 1040 | put_disk(disk); |
| 986 | module_put(owner); | 1041 | module_put(owner); |
| @@ -1079,63 +1134,6 @@ static int blkdev_open(struct inode * inode, struct file * filp) | |||
| 1079 | return res; | 1134 | return res; |
| 1080 | } | 1135 | } |
| 1081 | 1136 | ||
| 1082 | static int __blkdev_put(struct block_device *bdev, unsigned int subclass) | ||
| 1083 | { | ||
| 1084 | int ret = 0; | ||
| 1085 | struct inode *bd_inode = bdev->bd_inode; | ||
| 1086 | struct gendisk *disk = bdev->bd_disk; | ||
| 1087 | |||
| 1088 | mutex_lock_nested(&bdev->bd_mutex, subclass); | ||
| 1089 | lock_kernel(); | ||
| 1090 | if (!--bdev->bd_openers) { | ||
| 1091 | sync_blockdev(bdev); | ||
| 1092 | kill_bdev(bdev); | ||
| 1093 | } | ||
| 1094 | if (bdev->bd_contains == bdev) { | ||
| 1095 | if (disk->fops->release) | ||
| 1096 | ret = disk->fops->release(bd_inode, NULL); | ||
| 1097 | } else { | ||
| 1098 | mutex_lock_nested(&bdev->bd_contains->bd_mutex, | ||
| 1099 | subclass + 1); | ||
| 1100 | bdev->bd_contains->bd_part_count--; | ||
| 1101 | mutex_unlock(&bdev->bd_contains->bd_mutex); | ||
| 1102 | } | ||
| 1103 | if (!bdev->bd_openers) { | ||
| 1104 | struct module *owner = disk->fops->owner; | ||
| 1105 | |||
| 1106 | put_disk(disk); | ||
| 1107 | module_put(owner); | ||
| 1108 | |||
| 1109 | if (bdev->bd_contains != bdev) { | ||
| 1110 | kobject_put(&bdev->bd_part->kobj); | ||
| 1111 | bdev->bd_part = NULL; | ||
| 1112 | } | ||
| 1113 | bdev->bd_disk = NULL; | ||
| 1114 | bdev->bd_inode->i_data.backing_dev_info = &default_backing_dev_info; | ||
| 1115 | if (bdev != bdev->bd_contains) | ||
| 1116 | __blkdev_put(bdev->bd_contains, subclass + 1); | ||
| 1117 | bdev->bd_contains = NULL; | ||
| 1118 | } | ||
| 1119 | unlock_kernel(); | ||
| 1120 | mutex_unlock(&bdev->bd_mutex); | ||
| 1121 | bdput(bdev); | ||
| 1122 | return ret; | ||
| 1123 | } | ||
| 1124 | |||
| 1125 | int blkdev_put(struct block_device *bdev) | ||
| 1126 | { | ||
| 1127 | return __blkdev_put(bdev, BD_MUTEX_NORMAL); | ||
| 1128 | } | ||
| 1129 | |||
| 1130 | EXPORT_SYMBOL(blkdev_put); | ||
| 1131 | |||
| 1132 | int blkdev_put_partition(struct block_device *bdev) | ||
| 1133 | { | ||
| 1134 | return __blkdev_put(bdev, BD_MUTEX_PARTITION); | ||
| 1135 | } | ||
| 1136 | |||
| 1137 | EXPORT_SYMBOL(blkdev_put_partition); | ||
| 1138 | |||
| 1139 | static int blkdev_close(struct inode * inode, struct file * filp) | 1137 | static int blkdev_close(struct inode * inode, struct file * filp) |
| 1140 | { | 1138 | { |
| 1141 | struct block_device *bdev = I_BDEV(filp->f_mapping->host); | 1139 | struct block_device *bdev = I_BDEV(filp->f_mapping->host); |
