diff options
Diffstat (limited to 'fs')
-rw-r--r-- | fs/block_dev.c | 2 | ||||
-rw-r--r-- | fs/buffer.c | 74 | ||||
-rw-r--r-- | fs/ioctl.c | 46 |
3 files changed, 113 insertions, 9 deletions
diff --git a/fs/block_dev.c b/fs/block_dev.c index ac7031f12ea5..b3c1efff5e1d 100644 --- a/fs/block_dev.c +++ b/fs/block_dev.c | |||
@@ -285,6 +285,8 @@ static void init_once(void *foo) | |||
285 | INIT_LIST_HEAD(&bdev->bd_holder_list); | 285 | INIT_LIST_HEAD(&bdev->bd_holder_list); |
286 | #endif | 286 | #endif |
287 | inode_init_once(&ei->vfs_inode); | 287 | inode_init_once(&ei->vfs_inode); |
288 | /* Initialize mutex for freeze. */ | ||
289 | mutex_init(&bdev->bd_fsfreeze_mutex); | ||
288 | } | 290 | } |
289 | 291 | ||
290 | static inline void __bd_forget(struct inode *inode) | 292 | static inline void __bd_forget(struct inode *inode) |
diff --git a/fs/buffer.c b/fs/buffer.c index 87f9e537b8c3..b6e8b8632e2f 100644 --- a/fs/buffer.c +++ b/fs/buffer.c | |||
@@ -203,10 +203,25 @@ int fsync_bdev(struct block_device *bdev) | |||
203 | * happen on bdev until thaw_bdev() is called. | 203 | * happen on bdev until thaw_bdev() is called. |
204 | * If a superblock is found on this device, we take the s_umount semaphore | 204 | * If a superblock is found on this device, we take the s_umount semaphore |
205 | * on it to make sure nobody unmounts until the snapshot creation is done. | 205 | * on it to make sure nobody unmounts until the snapshot creation is done. |
206 | * The reference counter (bd_fsfreeze_count) guarantees that only the last | ||
207 | * unfreeze process can unfreeze the frozen filesystem actually when multiple | ||
208 | * freeze requests arrive simultaneously. It counts up in freeze_bdev() and | ||
209 | * count down in thaw_bdev(). When it becomes 0, thaw_bdev() will unfreeze | ||
210 | * actually. | ||
206 | */ | 211 | */ |
207 | struct super_block *freeze_bdev(struct block_device *bdev) | 212 | struct super_block *freeze_bdev(struct block_device *bdev) |
208 | { | 213 | { |
209 | struct super_block *sb; | 214 | struct super_block *sb; |
215 | int error = 0; | ||
216 | |||
217 | mutex_lock(&bdev->bd_fsfreeze_mutex); | ||
218 | if (bdev->bd_fsfreeze_count > 0) { | ||
219 | bdev->bd_fsfreeze_count++; | ||
220 | sb = get_super(bdev); | ||
221 | mutex_unlock(&bdev->bd_fsfreeze_mutex); | ||
222 | return sb; | ||
223 | } | ||
224 | bdev->bd_fsfreeze_count++; | ||
210 | 225 | ||
211 | down(&bdev->bd_mount_sem); | 226 | down(&bdev->bd_mount_sem); |
212 | sb = get_super(bdev); | 227 | sb = get_super(bdev); |
@@ -221,11 +236,24 @@ struct super_block *freeze_bdev(struct block_device *bdev) | |||
221 | 236 | ||
222 | sync_blockdev(sb->s_bdev); | 237 | sync_blockdev(sb->s_bdev); |
223 | 238 | ||
224 | if (sb->s_op->freeze_fs) | 239 | if (sb->s_op->freeze_fs) { |
225 | sb->s_op->freeze_fs(sb); | 240 | error = sb->s_op->freeze_fs(sb); |
241 | if (error) { | ||
242 | printk(KERN_ERR | ||
243 | "VFS:Filesystem freeze failed\n"); | ||
244 | sb->s_frozen = SB_UNFROZEN; | ||
245 | drop_super(sb); | ||
246 | up(&bdev->bd_mount_sem); | ||
247 | bdev->bd_fsfreeze_count--; | ||
248 | mutex_unlock(&bdev->bd_fsfreeze_mutex); | ||
249 | return ERR_PTR(error); | ||
250 | } | ||
251 | } | ||
226 | } | 252 | } |
227 | 253 | ||
228 | sync_blockdev(bdev); | 254 | sync_blockdev(bdev); |
255 | mutex_unlock(&bdev->bd_fsfreeze_mutex); | ||
256 | |||
229 | return sb; /* thaw_bdev releases s->s_umount and bd_mount_sem */ | 257 | return sb; /* thaw_bdev releases s->s_umount and bd_mount_sem */ |
230 | } | 258 | } |
231 | EXPORT_SYMBOL(freeze_bdev); | 259 | EXPORT_SYMBOL(freeze_bdev); |
@@ -237,20 +265,48 @@ EXPORT_SYMBOL(freeze_bdev); | |||
237 | * | 265 | * |
238 | * Unlocks the filesystem and marks it writeable again after freeze_bdev(). | 266 | * Unlocks the filesystem and marks it writeable again after freeze_bdev(). |
239 | */ | 267 | */ |
240 | void thaw_bdev(struct block_device *bdev, struct super_block *sb) | 268 | int thaw_bdev(struct block_device *bdev, struct super_block *sb) |
241 | { | 269 | { |
270 | int error = 0; | ||
271 | |||
272 | mutex_lock(&bdev->bd_fsfreeze_mutex); | ||
273 | if (!bdev->bd_fsfreeze_count) { | ||
274 | mutex_unlock(&bdev->bd_fsfreeze_mutex); | ||
275 | return -EINVAL; | ||
276 | } | ||
277 | |||
278 | bdev->bd_fsfreeze_count--; | ||
279 | if (bdev->bd_fsfreeze_count > 0) { | ||
280 | if (sb) | ||
281 | drop_super(sb); | ||
282 | mutex_unlock(&bdev->bd_fsfreeze_mutex); | ||
283 | return 0; | ||
284 | } | ||
285 | |||
242 | if (sb) { | 286 | if (sb) { |
243 | BUG_ON(sb->s_bdev != bdev); | 287 | BUG_ON(sb->s_bdev != bdev); |
244 | 288 | if (!(sb->s_flags & MS_RDONLY)) { | |
245 | if (sb->s_op->unfreeze_fs) | 289 | if (sb->s_op->unfreeze_fs) { |
246 | sb->s_op->unfreeze_fs(sb); | 290 | error = sb->s_op->unfreeze_fs(sb); |
247 | sb->s_frozen = SB_UNFROZEN; | 291 | if (error) { |
248 | smp_wmb(); | 292 | printk(KERN_ERR |
249 | wake_up(&sb->s_wait_unfrozen); | 293 | "VFS:Filesystem thaw failed\n"); |
294 | sb->s_frozen = SB_FREEZE_TRANS; | ||
295 | bdev->bd_fsfreeze_count++; | ||
296 | mutex_unlock(&bdev->bd_fsfreeze_mutex); | ||
297 | return error; | ||
298 | } | ||
299 | } | ||
300 | sb->s_frozen = SB_UNFROZEN; | ||
301 | smp_wmb(); | ||
302 | wake_up(&sb->s_wait_unfrozen); | ||
303 | } | ||
250 | drop_super(sb); | 304 | drop_super(sb); |
251 | } | 305 | } |
252 | 306 | ||
253 | up(&bdev->bd_mount_sem); | 307 | up(&bdev->bd_mount_sem); |
308 | mutex_unlock(&bdev->bd_fsfreeze_mutex); | ||
309 | return 0; | ||
254 | } | 310 | } |
255 | EXPORT_SYMBOL(thaw_bdev); | 311 | EXPORT_SYMBOL(thaw_bdev); |
256 | 312 | ||
diff --git a/fs/ioctl.c b/fs/ioctl.c index cc3f1aa1cf7b..20b0a8a24c6b 100644 --- a/fs/ioctl.c +++ b/fs/ioctl.c | |||
@@ -439,6 +439,43 @@ static int ioctl_fioasync(unsigned int fd, struct file *filp, | |||
439 | return error; | 439 | return error; |
440 | } | 440 | } |
441 | 441 | ||
442 | static int ioctl_fsfreeze(struct file *filp) | ||
443 | { | ||
444 | struct super_block *sb = filp->f_path.dentry->d_inode->i_sb; | ||
445 | |||
446 | if (!capable(CAP_SYS_ADMIN)) | ||
447 | return -EPERM; | ||
448 | |||
449 | /* If filesystem doesn't support freeze feature, return. */ | ||
450 | if (sb->s_op->freeze_fs == NULL) | ||
451 | return -EOPNOTSUPP; | ||
452 | |||
453 | /* If a blockdevice-backed filesystem isn't specified, return. */ | ||
454 | if (sb->s_bdev == NULL) | ||
455 | return -EINVAL; | ||
456 | |||
457 | /* Freeze */ | ||
458 | sb = freeze_bdev(sb->s_bdev); | ||
459 | if (IS_ERR(sb)) | ||
460 | return PTR_ERR(sb); | ||
461 | return 0; | ||
462 | } | ||
463 | |||
464 | static int ioctl_fsthaw(struct file *filp) | ||
465 | { | ||
466 | struct super_block *sb = filp->f_path.dentry->d_inode->i_sb; | ||
467 | |||
468 | if (!capable(CAP_SYS_ADMIN)) | ||
469 | return -EPERM; | ||
470 | |||
471 | /* If a blockdevice-backed filesystem isn't specified, return EINVAL. */ | ||
472 | if (sb->s_bdev == NULL) | ||
473 | return -EINVAL; | ||
474 | |||
475 | /* Thaw */ | ||
476 | return thaw_bdev(sb->s_bdev, sb); | ||
477 | } | ||
478 | |||
442 | /* | 479 | /* |
443 | * When you add any new common ioctls to the switches above and below | 480 | * When you add any new common ioctls to the switches above and below |
444 | * please update compat_sys_ioctl() too. | 481 | * please update compat_sys_ioctl() too. |
@@ -486,6 +523,15 @@ int do_vfs_ioctl(struct file *filp, unsigned int fd, unsigned int cmd, | |||
486 | } else | 523 | } else |
487 | error = -ENOTTY; | 524 | error = -ENOTTY; |
488 | break; | 525 | break; |
526 | |||
527 | case FIFREEZE: | ||
528 | error = ioctl_fsfreeze(filp); | ||
529 | break; | ||
530 | |||
531 | case FITHAW: | ||
532 | error = ioctl_fsthaw(filp); | ||
533 | break; | ||
534 | |||
489 | default: | 535 | default: |
490 | if (S_ISREG(filp->f_path.dentry->d_inode->i_mode)) | 536 | if (S_ISREG(filp->f_path.dentry->d_inode->i_mode)) |
491 | error = file_ioctl(filp, cmd, arg); | 537 | error = file_ioctl(filp, cmd, arg); |