aboutsummaryrefslogtreecommitdiffstats
path: root/fs/btrfs/inode.c
diff options
context:
space:
mode:
Diffstat (limited to 'fs/btrfs/inode.c')
-rw-r--r--fs/btrfs/inode.c49
1 files changed, 38 insertions, 11 deletions
diff --git a/fs/btrfs/inode.c b/fs/btrfs/inode.c
index 6402a41b9023..fe3891e240b3 100644
--- a/fs/btrfs/inode.c
+++ b/fs/btrfs/inode.c
@@ -3527,6 +3527,8 @@ void btrfs_evict_inode(struct inode *inode)
3527{ 3527{
3528 struct btrfs_trans_handle *trans; 3528 struct btrfs_trans_handle *trans;
3529 struct btrfs_root *root = BTRFS_I(inode)->root; 3529 struct btrfs_root *root = BTRFS_I(inode)->root;
3530 struct btrfs_block_rsv *rsv;
3531 u64 min_size = btrfs_calc_trans_metadata_size(root, 2);
3530 unsigned long nr; 3532 unsigned long nr;
3531 int ret; 3533 int ret;
3532 3534
@@ -3554,22 +3556,44 @@ void btrfs_evict_inode(struct inode *inode)
3554 goto no_delete; 3556 goto no_delete;
3555 } 3557 }
3556 3558
3559 rsv = btrfs_alloc_block_rsv(root);
3560 if (!rsv) {
3561 btrfs_orphan_del(NULL, inode);
3562 goto no_delete;
3563 }
3564
3557 btrfs_i_size_write(inode, 0); 3565 btrfs_i_size_write(inode, 0);
3558 3566
3567 /*
3568 * This is a bit simpler than btrfs_truncate since
3569 *
3570 * 1) We've already reserved our space for our orphan item in the
3571 * unlink.
3572 * 2) We're going to delete the inode item, so we don't need to update
3573 * it at all.
3574 *
3575 * So we just need to reserve some slack space in case we add bytes when
3576 * doing the truncate.
3577 */
3559 while (1) { 3578 while (1) {
3560 trans = btrfs_join_transaction(root); 3579 ret = btrfs_block_rsv_check(NULL, root, rsv, min_size, 0);
3561 BUG_ON(IS_ERR(trans));
3562 trans->block_rsv = root->orphan_block_rsv;
3563
3564 ret = btrfs_block_rsv_check(trans, root,
3565 root->orphan_block_rsv, 0, 5);
3566 if (ret) { 3580 if (ret) {
3567 BUG_ON(ret != -EAGAIN); 3581 printk(KERN_WARNING "Could not get space for a "
3568 ret = btrfs_commit_transaction(trans, root); 3582 "delete, will truncate on mount\n");
3569 BUG_ON(ret); 3583 btrfs_orphan_del(NULL, inode);
3570 continue; 3584 btrfs_free_block_rsv(root, rsv);
3585 goto no_delete;
3586 }
3587
3588 trans = btrfs_start_transaction(root, 0);
3589 if (IS_ERR(trans)) {
3590 btrfs_orphan_del(NULL, inode);
3591 btrfs_free_block_rsv(root, rsv);
3592 goto no_delete;
3571 } 3593 }
3572 3594
3595 trans->block_rsv = rsv;
3596
3573 ret = btrfs_truncate_inode_items(trans, root, inode, 0, 0); 3597 ret = btrfs_truncate_inode_items(trans, root, inode, 0, 0);
3574 if (ret != -EAGAIN) 3598 if (ret != -EAGAIN)
3575 break; 3599 break;
@@ -3578,14 +3602,17 @@ void btrfs_evict_inode(struct inode *inode)
3578 btrfs_end_transaction(trans, root); 3602 btrfs_end_transaction(trans, root);
3579 trans = NULL; 3603 trans = NULL;
3580 btrfs_btree_balance_dirty(root, nr); 3604 btrfs_btree_balance_dirty(root, nr);
3581
3582 } 3605 }
3583 3606
3607 btrfs_free_block_rsv(root, rsv);
3608
3584 if (ret == 0) { 3609 if (ret == 0) {
3610 trans->block_rsv = root->orphan_block_rsv;
3585 ret = btrfs_orphan_del(trans, inode); 3611 ret = btrfs_orphan_del(trans, inode);
3586 BUG_ON(ret); 3612 BUG_ON(ret);
3587 } 3613 }
3588 3614
3615 trans->block_rsv = &root->fs_info->trans_block_rsv;
3589 if (!(root == root->fs_info->tree_root || 3616 if (!(root == root->fs_info->tree_root ||
3590 root->root_key.objectid == BTRFS_TREE_RELOC_OBJECTID)) 3617 root->root_key.objectid == BTRFS_TREE_RELOC_OBJECTID))
3591 btrfs_return_ino(root, btrfs_ino(inode)); 3618 btrfs_return_ino(root, btrfs_ino(inode));