diff options
Diffstat (limited to 'fs/btrfs/extent_io.c')
-rw-r--r-- | fs/btrfs/extent_io.c | 56 |
1 files changed, 40 insertions, 16 deletions
diff --git a/fs/btrfs/extent_io.c b/fs/btrfs/extent_io.c index 64c8b361b539..a90c4a12556b 100644 --- a/fs/btrfs/extent_io.c +++ b/fs/btrfs/extent_io.c | |||
@@ -439,6 +439,15 @@ static int clear_state_bit(struct extent_io_tree *tree, | |||
439 | return ret; | 439 | return ret; |
440 | } | 440 | } |
441 | 441 | ||
442 | static struct extent_state * | ||
443 | alloc_extent_state_atomic(struct extent_state *prealloc) | ||
444 | { | ||
445 | if (!prealloc) | ||
446 | prealloc = alloc_extent_state(GFP_ATOMIC); | ||
447 | |||
448 | return prealloc; | ||
449 | } | ||
450 | |||
442 | /* | 451 | /* |
443 | * clear some bits on a range in the tree. This may require splitting | 452 | * clear some bits on a range in the tree. This may require splitting |
444 | * or inserting elements in the tree, so the gfp mask is used to | 453 | * or inserting elements in the tree, so the gfp mask is used to |
@@ -476,8 +485,7 @@ int clear_extent_bit(struct extent_io_tree *tree, u64 start, u64 end, | |||
476 | again: | 485 | again: |
477 | if (!prealloc && (mask & __GFP_WAIT)) { | 486 | if (!prealloc && (mask & __GFP_WAIT)) { |
478 | prealloc = alloc_extent_state(mask); | 487 | prealloc = alloc_extent_state(mask); |
479 | if (!prealloc) | 488 | BUG_ON(!prealloc); |
480 | return -ENOMEM; | ||
481 | } | 489 | } |
482 | 490 | ||
483 | spin_lock(&tree->lock); | 491 | spin_lock(&tree->lock); |
@@ -529,8 +537,8 @@ hit_next: | |||
529 | */ | 537 | */ |
530 | 538 | ||
531 | if (state->start < start) { | 539 | if (state->start < start) { |
532 | if (!prealloc) | 540 | prealloc = alloc_extent_state_atomic(prealloc); |
533 | prealloc = alloc_extent_state(GFP_ATOMIC); | 541 | BUG_ON(!prealloc); |
534 | err = split_state(tree, state, prealloc, start); | 542 | err = split_state(tree, state, prealloc, start); |
535 | BUG_ON(err == -EEXIST); | 543 | BUG_ON(err == -EEXIST); |
536 | prealloc = NULL; | 544 | prealloc = NULL; |
@@ -551,8 +559,8 @@ hit_next: | |||
551 | * on the first half | 559 | * on the first half |
552 | */ | 560 | */ |
553 | if (state->start <= end && state->end > end) { | 561 | if (state->start <= end && state->end > end) { |
554 | if (!prealloc) | 562 | prealloc = alloc_extent_state_atomic(prealloc); |
555 | prealloc = alloc_extent_state(GFP_ATOMIC); | 563 | BUG_ON(!prealloc); |
556 | err = split_state(tree, state, prealloc, end + 1); | 564 | err = split_state(tree, state, prealloc, end + 1); |
557 | BUG_ON(err == -EEXIST); | 565 | BUG_ON(err == -EEXIST); |
558 | if (wake) | 566 | if (wake) |
@@ -725,8 +733,7 @@ int set_extent_bit(struct extent_io_tree *tree, u64 start, u64 end, | |||
725 | again: | 733 | again: |
726 | if (!prealloc && (mask & __GFP_WAIT)) { | 734 | if (!prealloc && (mask & __GFP_WAIT)) { |
727 | prealloc = alloc_extent_state(mask); | 735 | prealloc = alloc_extent_state(mask); |
728 | if (!prealloc) | 736 | BUG_ON(!prealloc); |
729 | return -ENOMEM; | ||
730 | } | 737 | } |
731 | 738 | ||
732 | spin_lock(&tree->lock); | 739 | spin_lock(&tree->lock); |
@@ -743,6 +750,8 @@ again: | |||
743 | */ | 750 | */ |
744 | node = tree_search(tree, start); | 751 | node = tree_search(tree, start); |
745 | if (!node) { | 752 | if (!node) { |
753 | prealloc = alloc_extent_state_atomic(prealloc); | ||
754 | BUG_ON(!prealloc); | ||
746 | err = insert_state(tree, prealloc, start, end, &bits); | 755 | err = insert_state(tree, prealloc, start, end, &bits); |
747 | prealloc = NULL; | 756 | prealloc = NULL; |
748 | BUG_ON(err == -EEXIST); | 757 | BUG_ON(err == -EEXIST); |
@@ -771,20 +780,18 @@ hit_next: | |||
771 | if (err) | 780 | if (err) |
772 | goto out; | 781 | goto out; |
773 | 782 | ||
783 | next_node = rb_next(node); | ||
774 | cache_state(state, cached_state); | 784 | cache_state(state, cached_state); |
775 | merge_state(tree, state); | 785 | merge_state(tree, state); |
776 | if (last_end == (u64)-1) | 786 | if (last_end == (u64)-1) |
777 | goto out; | 787 | goto out; |
778 | 788 | ||
779 | start = last_end + 1; | 789 | start = last_end + 1; |
780 | if (start < end && prealloc && !need_resched()) { | 790 | if (next_node && start < end && prealloc && !need_resched()) { |
781 | next_node = rb_next(node); | 791 | state = rb_entry(next_node, struct extent_state, |
782 | if (next_node) { | 792 | rb_node); |
783 | state = rb_entry(next_node, struct extent_state, | 793 | if (state->start == start) |
784 | rb_node); | 794 | goto hit_next; |
785 | if (state->start == start) | ||
786 | goto hit_next; | ||
787 | } | ||
788 | } | 795 | } |
789 | goto search_again; | 796 | goto search_again; |
790 | } | 797 | } |
@@ -811,6 +818,9 @@ hit_next: | |||
811 | err = -EEXIST; | 818 | err = -EEXIST; |
812 | goto out; | 819 | goto out; |
813 | } | 820 | } |
821 | |||
822 | prealloc = alloc_extent_state_atomic(prealloc); | ||
823 | BUG_ON(!prealloc); | ||
814 | err = split_state(tree, state, prealloc, start); | 824 | err = split_state(tree, state, prealloc, start); |
815 | BUG_ON(err == -EEXIST); | 825 | BUG_ON(err == -EEXIST); |
816 | prealloc = NULL; | 826 | prealloc = NULL; |
@@ -841,14 +851,25 @@ hit_next: | |||
841 | this_end = end; | 851 | this_end = end; |
842 | else | 852 | else |
843 | this_end = last_start - 1; | 853 | this_end = last_start - 1; |
854 | |||
855 | prealloc = alloc_extent_state_atomic(prealloc); | ||
856 | BUG_ON(!prealloc); | ||
857 | |||
858 | /* | ||
859 | * Avoid to free 'prealloc' if it can be merged with | ||
860 | * the later extent. | ||
861 | */ | ||
862 | atomic_inc(&prealloc->refs); | ||
844 | err = insert_state(tree, prealloc, start, this_end, | 863 | err = insert_state(tree, prealloc, start, this_end, |
845 | &bits); | 864 | &bits); |
846 | BUG_ON(err == -EEXIST); | 865 | BUG_ON(err == -EEXIST); |
847 | if (err) { | 866 | if (err) { |
867 | free_extent_state(prealloc); | ||
848 | prealloc = NULL; | 868 | prealloc = NULL; |
849 | goto out; | 869 | goto out; |
850 | } | 870 | } |
851 | cache_state(prealloc, cached_state); | 871 | cache_state(prealloc, cached_state); |
872 | free_extent_state(prealloc); | ||
852 | prealloc = NULL; | 873 | prealloc = NULL; |
853 | start = this_end + 1; | 874 | start = this_end + 1; |
854 | goto search_again; | 875 | goto search_again; |
@@ -865,6 +886,9 @@ hit_next: | |||
865 | err = -EEXIST; | 886 | err = -EEXIST; |
866 | goto out; | 887 | goto out; |
867 | } | 888 | } |
889 | |||
890 | prealloc = alloc_extent_state_atomic(prealloc); | ||
891 | BUG_ON(!prealloc); | ||
868 | err = split_state(tree, state, prealloc, end + 1); | 892 | err = split_state(tree, state, prealloc, end + 1); |
869 | BUG_ON(err == -EEXIST); | 893 | BUG_ON(err == -EEXIST); |
870 | 894 | ||