found->total_bytes += total_bytes;
                found->bytes_used += bytes_used;
                found->full = 0;
-               WARN_ON(found->total_bytes < found->bytes_used);
                *space_info = found;
                return 0;
        }
                 */
                clear_page_dirty_for_io(page);
 
-               set_extent_delalloc(io_tree, page_start,
-                                   page_end, GFP_NOFS);
+               btrfs_set_extent_delalloc(inode, page_start, page_end);
                set_page_dirty(page);
 
                unlock_extent(io_tree, page_start, page_end, GFP_NOFS);
        key.type = 0;
        cur_byte = key.objectid;
 
+       mutex_unlock(&root->fs_info->alloc_mutex);
+
+       btrfs_start_delalloc_inodes(root);
+       btrfs_wait_ordered_extents(tree_root);
+
+       mutex_lock(&root->fs_info->alloc_mutex);
+
        ret = btrfs_search_slot(NULL, root, &key, path, 0, 0);
        if (ret < 0)
                goto out;
 
                btrfs_clean_old_snapshots(tree_root);
 
+               btrfs_start_delalloc_inodes(root);
                btrfs_wait_ordered_extents(tree_root);
 
                trans = btrfs_start_transaction(tree_root, 1);
 
                spin_lock_irqsave(&root->fs_info->delalloc_lock, flags);
                BTRFS_I(inode)->delalloc_bytes += end - start + 1;
                root->fs_info->delalloc_bytes += end - start + 1;
+               if (list_empty(&BTRFS_I(inode)->delalloc_inodes)) {
+                       list_add_tail(&BTRFS_I(inode)->delalloc_inodes,
+                                     &root->fs_info->delalloc_inodes);
+               }
                spin_unlock_irqrestore(&root->fs_info->delalloc_lock, flags);
        }
        return 0;
                        root->fs_info->delalloc_bytes -= end - start + 1;
                        BTRFS_I(inode)->delalloc_bytes -= end - start + 1;
                }
+               if (BTRFS_I(inode)->delalloc_bytes == 0 &&
+                   !list_empty(&BTRFS_I(inode)->delalloc_inodes)) {
+                       list_del_init(&BTRFS_I(inode)->delalloc_inodes);
+               }
                spin_unlock_irqrestore(&root->fs_info->delalloc_lock, flags);
        }
        return 0;
        return 0;
 }
 
+int btrfs_set_extent_delalloc(struct inode *inode, u64 start, u64 end)
+{
+       return set_extent_delalloc(&BTRFS_I(inode)->io_tree, start, end,
+                                  GFP_NOFS);
+}
+
 struct btrfs_writepage_fixup {
        struct page *page;
        struct btrfs_work work;
                goto again;
        }
 
-       set_extent_delalloc(&BTRFS_I(inode)->io_tree, page_start, page_end,
-                           GFP_NOFS);
+       btrfs_set_extent_delalloc(inode, page_start, page_end);
        ClearPageChecked(page);
 out:
        unlock_extent(&BTRFS_I(inode)->io_tree, page_start, page_end, GFP_NOFS);
                goto again;
        }
 
-       set_extent_delalloc(&BTRFS_I(inode)->io_tree, page_start,
-                           page_end, GFP_NOFS);
+       btrfs_set_extent_delalloc(inode, page_start, page_end);
        ret = 0;
        if (offset != PAGE_CACHE_SIZE) {
                kaddr = kmap(page);
                             inode->i_mapping, GFP_NOFS);
        extent_io_tree_init(&BTRFS_I(inode)->io_failure_tree,
                             inode->i_mapping, GFP_NOFS);
+       INIT_LIST_HEAD(&BTRFS_I(inode)->delalloc_inodes);
        btrfs_ordered_inode_tree_init(&BTRFS_I(inode)->ordered_tree);
        mutex_init(&BTRFS_I(inode)->csum_mutex);
        mutex_init(&BTRFS_I(inode)->extent_mutex);
        extent_io_tree_init(&BTRFS_I(inode)->io_failure_tree,
                             inode->i_mapping, GFP_NOFS);
        btrfs_ordered_inode_tree_init(&BTRFS_I(inode)->ordered_tree);
+       INIT_LIST_HEAD(&BTRFS_I(inode)->delalloc_inodes);
        mutex_init(&BTRFS_I(inode)->csum_mutex);
        mutex_init(&BTRFS_I(inode)->extent_mutex);
        BTRFS_I(inode)->delalloc_bytes = 0;
                                     inode->i_mapping, GFP_NOFS);
                extent_io_tree_init(&BTRFS_I(inode)->io_failure_tree,
                                     inode->i_mapping, GFP_NOFS);
+               INIT_LIST_HEAD(&BTRFS_I(inode)->delalloc_inodes);
                mutex_init(&BTRFS_I(inode)->csum_mutex);
                mutex_init(&BTRFS_I(inode)->extent_mutex);
                BTRFS_I(inode)->delalloc_bytes = 0;
                goto again;
        }
 
-       set_extent_delalloc(&BTRFS_I(inode)->io_tree, page_start,
-                           page_end, GFP_NOFS);
+       btrfs_set_extent_delalloc(inode, page_start, page_end);
        ret = 0;
 
        /* page is wholly or partially inside EOF */
        return ret;
 }
 
+int btrfs_start_delalloc_inodes(struct btrfs_root *root)
+{
+       struct list_head *head = &root->fs_info->delalloc_inodes;
+       struct btrfs_inode *binode;
+       unsigned long flags;
+
+       spin_lock_irqsave(&root->fs_info->delalloc_lock, flags);
+       while(!list_empty(head)) {
+               binode = list_entry(head->next, struct btrfs_inode,
+                                   delalloc_inodes);
+               atomic_inc(&binode->vfs_inode.i_count);
+               spin_unlock_irqrestore(&root->fs_info->delalloc_lock, flags);
+               filemap_write_and_wait(binode->vfs_inode.i_mapping);
+               iput(&binode->vfs_inode);
+               spin_lock_irqsave(&root->fs_info->delalloc_lock, flags);
+       }
+       spin_unlock_irqrestore(&root->fs_info->delalloc_lock, flags);
+       return 0;
+}
+
 static int btrfs_symlink(struct inode *dir, struct dentry *dentry,
                         const char *symname)
 {
                                     inode->i_mapping, GFP_NOFS);
                extent_io_tree_init(&BTRFS_I(inode)->io_failure_tree,
                                     inode->i_mapping, GFP_NOFS);
+               INIT_LIST_HEAD(&BTRFS_I(inode)->delalloc_inodes);
                mutex_init(&BTRFS_I(inode)->csum_mutex);
                mutex_init(&BTRFS_I(inode)->extent_mutex);
                BTRFS_I(inode)->delalloc_bytes = 0;