disk-io.c 18.3 KB
Newer Older
C
Chris Mason 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
/*
 * Copyright (C) 2007 Oracle.  All rights reserved.
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public
 * License v2 as published by the Free Software Foundation.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * General Public License for more details.
 *
 * You should have received a copy of the GNU General Public
 * License along with this program; if not, write to the
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
 * Boston, MA 021110-1307, USA.
 */

C
Chris Mason 已提交
19
#include <linux/fs.h>
20
#include <linux/blkdev.h>
21
#include <linux/crc32c.h>
C
Chris Mason 已提交
22
#include <linux/scatterlist.h>
C
Chris Mason 已提交
23
#include <linux/swap.h>
24
#include <linux/radix-tree.h>
C
Chris Mason 已提交
25
#include <linux/writeback.h>
26
#include <linux/buffer_head.h> // for block_sync_page
27 28
#include "ctree.h"
#include "disk-io.h"
29
#include "transaction.h"
30
#include "btrfs_inode.h"
31
#include "print-tree.h"
32

33 34
#if 0
static int check_tree_block(struct btrfs_root *root, struct extent_buffer *buf)
35
{
36 37 38 39
	if (extent_buffer_blocknr(buf) != btrfs_header_blocknr(buf)) {
		printk(KERN_CRIT "buf blocknr(buf) is %llu, header is %llu\n",
		       (unsigned long long)extent_buffer_blocknr(buf),
		       (unsigned long long)btrfs_header_blocknr(buf));
C
Chris Mason 已提交
40
		return 1;
41
	}
42
	return 0;
43
}
44
#endif
45

46
struct extent_buffer *btrfs_find_tree_block(struct btrfs_root *root,
47
					    u64 bytenr, u32 blocksize)
48
{
49
	struct inode *btree_inode = root->fs_info->btree_inode;
50 51
	struct extent_buffer *eb;
	eb = find_extent_buffer(&BTRFS_I(btree_inode)->extent_tree,
52
				bytenr, blocksize, GFP_NOFS);
53 54 55
	if (eb)
		eb->alloc_addr = (unsigned long)__builtin_return_address(0);
	return eb;
56
}
57

58
struct extent_buffer *btrfs_find_create_tree_block(struct btrfs_root *root,
59
						 u64 bytenr, u32 blocksize)
60 61
{
	struct inode *btree_inode = root->fs_info->btree_inode;
62
	struct extent_buffer *eb;
63

64
	eb = alloc_extent_buffer(&BTRFS_I(btree_inode)->extent_tree,
65
				 bytenr, blocksize, GFP_NOFS);
66 67
	eb->alloc_addr = (unsigned long)__builtin_return_address(0);
	return eb;
68 69
}

70 71 72
struct extent_map *btree_get_extent(struct inode *inode, struct page *page,
				    size_t page_offset, u64 start, u64 end,
				    int create)
73
{
74 75 76 77 78 79 80 81
	struct extent_map_tree *em_tree = &BTRFS_I(inode)->extent_tree;
	struct extent_map *em;
	int ret;

again:
	em = lookup_extent_mapping(em_tree, start, end);
	if (em) {
		goto out;
82
	}
83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102
	em = alloc_extent_map(GFP_NOFS);
	if (!em) {
		em = ERR_PTR(-ENOMEM);
		goto out;
	}
	em->start = 0;
	em->end = (i_size_read(inode) & ~((u64)PAGE_CACHE_SIZE -1)) - 1;
	em->block_start = 0;
	em->block_end = em->end;
	em->bdev = inode->i_sb->s_bdev;
	ret = add_extent_mapping(em_tree, em);
	if (ret == -EEXIST) {
		free_extent_map(em);
		em = NULL;
		goto again;
	} else if (ret) {
		em = ERR_PTR(ret);
	}
out:
	return em;
103 104
}

105
static int btree_writepage(struct page *page, struct writeback_control *wbc)
106
{
107 108 109 110 111 112 113 114 115 116
	struct extent_map_tree *tree;
	tree = &BTRFS_I(page->mapping->host)->extent_tree;
	return extent_write_full_page(tree, page, btree_get_extent, wbc);
}
int btree_readpage(struct file *file, struct page *page)
{
	struct extent_map_tree *tree;
	tree = &BTRFS_I(page->mapping->host)->extent_tree;
	return extent_read_full_page(tree, page, btree_get_extent);
}
C
Chris Mason 已提交
117

118 119 120 121
static int btree_releasepage(struct page *page, gfp_t unused_gfp_flags)
{
	struct extent_map_tree *tree;
	int ret;
122

123 124 125 126 127 128 129 130
	BUG_ON(page->private != 1);
	tree = &BTRFS_I(page->mapping->host)->extent_tree;
	ret = try_release_extent_mapping(tree, page);
	if (ret == 1) {
		ClearPagePrivate(page);
		set_page_private(page, 0);
		page_cache_release(page);
	}
131 132 133
	return ret;
}

134
static void btree_invalidatepage(struct page *page, unsigned long offset)
135
{
136 137 138 139
	struct extent_map_tree *tree;
	tree = &BTRFS_I(page->mapping->host)->extent_tree;
	extent_invalidatepage(tree, page, offset);
	btree_releasepage(page, GFP_NOFS);
140 141
}

C
Chris Mason 已提交
142 143
int btrfs_csum_data(struct btrfs_root * root, char *data, size_t len,
		    char *result)
C
Chris Mason 已提交
144
{
145 146
	return 0;
#if 0
147 148 149 150
	u32 crc;
	crc = crc32c(0, data, len);
	memcpy(result, &crc, BTRFS_CRC32_SIZE);
	return 0;
151
#endif
C
Chris Mason 已提交
152
}
153

154 155
#if 0
static int csum_tree_block(struct btrfs_root *root, struct extent_buffer *buf,
C
Chris Mason 已提交
156 157
			   int verify)
{
158
	return 0;
159
	char result[BTRFS_CRC32_SIZE];
C
Chris Mason 已提交
160 161 162 163 164 165 166
	int ret;
	struct btrfs_node *node;

	ret = btrfs_csum_data(root, bh->b_data + BTRFS_CSUM_SIZE,
			      bh->b_size - BTRFS_CSUM_SIZE, result);
	if (ret)
		return ret;
C
Chris Mason 已提交
167
	if (verify) {
168
		if (memcmp(bh->b_data, result, BTRFS_CRC32_SIZE)) {
C
Chris Mason 已提交
169 170 171
			printk("btrfs: %s checksum verify failed on %llu\n",
			       root->fs_info->sb->s_id,
			       (unsigned long long)bh_blocknr(bh));
C
Chris Mason 已提交
172 173 174 175
			return 1;
		}
	} else {
		node = btrfs_buffer_node(bh);
176
		memcpy(node->header.csum, result, BTRFS_CRC32_SIZE);
C
Chris Mason 已提交
177
	}
C
Chris Mason 已提交
178 179
	return 0;
}
180
#endif
C
Chris Mason 已提交
181

182
#if 0
183
static int btree_writepage(struct page *page, struct writeback_control *wbc)
184
{
C
Chris Mason 已提交
185
	struct buffer_head *bh;
186
	struct btrfs_root *root = BTRFS_I(page->mapping->host)->root;
C
Chris Mason 已提交
187 188 189 190 191 192 193 194 195 196 197 198
	struct buffer_head *head;
	if (!page_has_buffers(page)) {
		create_empty_buffers(page, root->fs_info->sb->s_blocksize,
					(1 << BH_Dirty)|(1 << BH_Uptodate));
	}
	head = page_buffers(page);
	bh = head;
	do {
		if (buffer_dirty(bh))
			csum_tree_block(root, bh, 0);
		bh = bh->b_this_page;
	} while (bh != head);
199
	return block_write_full_page(page, btree_get_block, wbc);
200
}
201
#endif
202

203 204 205
static struct address_space_operations btree_aops = {
	.readpage	= btree_readpage,
	.writepage	= btree_writepage,
206 207
	.releasepage	= btree_releasepage,
	.invalidatepage = btree_invalidatepage,
208 209 210
	.sync_page	= block_sync_page,
};

211
int readahead_tree_block(struct btrfs_root *root, u64 bytenr, u32 blocksize)
C
Chris Mason 已提交
212
{
213 214
	struct extent_buffer *buf = NULL;
	struct inode *btree_inode = root->fs_info->btree_inode;
215
	int ret = 0;
C
Chris Mason 已提交
216

217
	buf = btrfs_find_create_tree_block(root, bytenr, blocksize);
218
	if (!buf)
C
Chris Mason 已提交
219
		return 0;
220 221 222
	read_extent_buffer_pages(&BTRFS_I(btree_inode)->extent_tree,
				 buf, 0);
	free_extent_buffer(buf);
223
	return ret;
C
Chris Mason 已提交
224 225
}

226 227
struct extent_buffer *read_tree_block(struct btrfs_root *root, u64 bytenr,
				      u32 blocksize)
228
{
229 230 231
	struct extent_buffer *buf = NULL;
	struct inode *btree_inode = root->fs_info->btree_inode;

232
	buf = btrfs_find_create_tree_block(root, bytenr, blocksize);
233 234 235 236
	if (!buf)
		return NULL;
	read_extent_buffer_pages(&BTRFS_I(btree_inode)->extent_tree,
				 buf, 1);
237
	buf->alloc_addr = (unsigned long)__builtin_return_address(0);
238
	return buf;
239 240
}

241
int clean_tree_block(struct btrfs_trans_handle *trans, struct btrfs_root *root,
242
		     struct extent_buffer *buf)
243
{
244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261
	struct inode *btree_inode = root->fs_info->btree_inode;
	clear_extent_buffer_dirty(&BTRFS_I(btree_inode)->extent_tree, buf);
	return 0;
}

int wait_on_tree_block_writeback(struct btrfs_root *root,
				 struct extent_buffer *buf)
{
	struct inode *btree_inode = root->fs_info->btree_inode;
	wait_on_extent_buffer_writeback(&BTRFS_I(btree_inode)->extent_tree,
					buf);
	return 0;
}

int set_tree_block_dirty(struct btrfs_root *root, struct extent_buffer *buf)
{
	struct inode *btree_inode = root->fs_info->btree_inode;
	set_extent_buffer_dirty(&BTRFS_I(btree_inode)->extent_tree, buf);
262 263 264
	return 0;
}

265
static int __setup_root(u32 nodesize, u32 leafsize, u32 sectorsize,
266 267
			struct btrfs_root *root,
			struct btrfs_fs_info *fs_info,
C
Chris Mason 已提交
268
			u64 objectid)
269
{
C
Chris Mason 已提交
270
	root->node = NULL;
271
	root->inode = NULL;
272
	root->commit_root = NULL;
273 274 275
	root->sectorsize = sectorsize;
	root->nodesize = nodesize;
	root->leafsize = leafsize;
C
Chris Mason 已提交
276
	root->ref_cows = 0;
277
	root->fs_info = fs_info;
278 279
	root->objectid = objectid;
	root->last_trans = 0;
C
Chris Mason 已提交
280 281
	root->highest_inode = 0;
	root->last_inode_alloc = 0;
282
	root->name = NULL;
283 284
	memset(&root->root_key, 0, sizeof(root->root_key));
	memset(&root->root_item, 0, sizeof(root->root_item));
285
	memset(&root->defrag_progress, 0, sizeof(root->defrag_progress));
286 287
	memset(&root->root_kobj, 0, sizeof(root->root_kobj));
	init_completion(&root->kobj_unregister);
288
	init_rwsem(&root->snap_sem);
289 290
	root->defrag_running = 0;
	root->defrag_level = 0;
291
	root->root_key.objectid = objectid;
292 293 294
	return 0;
}

295
static int find_and_setup_root(struct btrfs_root *tree_root,
296 297
			       struct btrfs_fs_info *fs_info,
			       u64 objectid,
C
Chris Mason 已提交
298
			       struct btrfs_root *root)
299 300
{
	int ret;
301
	u32 blocksize;
302

303 304
	__setup_root(tree_root->nodesize, tree_root->leafsize,
		     tree_root->sectorsize, root, fs_info, objectid);
305 306 307 308
	ret = btrfs_find_last_root(tree_root, objectid,
				   &root->root_item, &root->root_key);
	BUG_ON(ret);

309 310 311
	blocksize = btrfs_level_size(root, btrfs_root_level(&root->root_item));
	root->node = read_tree_block(root, btrfs_root_bytenr(&root->root_item),
				     blocksize);
312
	BUG_ON(!root->node);
313 314 315
	return 0;
}

316 317
struct btrfs_root *btrfs_read_fs_root_no_radix(struct btrfs_fs_info *fs_info,
					       struct btrfs_key *location)
318 319 320 321
{
	struct btrfs_root *root;
	struct btrfs_root *tree_root = fs_info->tree_root;
	struct btrfs_path *path;
322
	struct extent_buffer *l;
C
Chris Mason 已提交
323
	u64 highest_inode;
324
	u32 blocksize;
325 326
	int ret = 0;

327
	root = kzalloc(sizeof(*root), GFP_NOFS);
C
Chris Mason 已提交
328
	if (!root)
329 330
		return ERR_PTR(-ENOMEM);
	if (location->offset == (u64)-1) {
331
		ret = find_and_setup_root(tree_root, fs_info,
332 333 334 335 336 337 338 339
					  location->objectid, root);
		if (ret) {
			kfree(root);
			return ERR_PTR(ret);
		}
		goto insert;
	}

340 341
	__setup_root(tree_root->nodesize, tree_root->leafsize,
		     tree_root->sectorsize, root, fs_info,
342 343 344 345 346 347 348 349 350 351
		     location->objectid);

	path = btrfs_alloc_path();
	BUG_ON(!path);
	ret = btrfs_search_slot(NULL, tree_root, location, path, 0, 0);
	if (ret != 0) {
		if (ret > 0)
			ret = -ENOENT;
		goto out;
	}
352 353 354
	l = path->nodes[0];
	read_extent_buffer(l, &root->root_item,
	       btrfs_item_ptr_offset(l, path->slots[0]),
355 356 357 358 359 360 361 362 363
	       sizeof(root->root_item));
	ret = 0;
out:
	btrfs_release_path(root, path);
	btrfs_free_path(path);
	if (ret) {
		kfree(root);
		return ERR_PTR(ret);
	}
364 365 366
	blocksize = btrfs_level_size(root, btrfs_root_level(&root->root_item));
	root->node = read_tree_block(root, btrfs_root_bytenr(&root->root_item),
				     blocksize);
367 368 369
	BUG_ON(!root->node);
insert:
	root->ref_cows = 1;
370 371 372 373 374 375 376 377 378
	ret = btrfs_find_highest_inode(root, &highest_inode);
	if (ret == 0) {
		root->highest_inode = highest_inode;
		root->last_inode_alloc = highest_inode;
	}
	return root;
}

struct btrfs_root *btrfs_read_fs_root(struct btrfs_fs_info *fs_info,
379 380
				      struct btrfs_key *location,
				      const char *name, int namelen)
381 382 383 384 385 386 387 388 389 390 391 392
{
	struct btrfs_root *root;
	int ret;

	root = radix_tree_lookup(&fs_info->fs_roots_radix,
				 (unsigned long)location->objectid);
	if (root)
		return root;

	root = btrfs_read_fs_root_no_radix(fs_info, location);
	if (IS_ERR(root))
		return root;
C
Chris Mason 已提交
393 394
	ret = radix_tree_insert(&fs_info->fs_roots_radix,
				(unsigned long)root->root_key.objectid,
395 396
				root);
	if (ret) {
397
		free_extent_buffer(root->node);
398 399 400
		kfree(root);
		return ERR_PTR(ret);
	}
401 402 403

	ret = btrfs_set_root_name(root, name, namelen);
	if (ret) {
404
		free_extent_buffer(root->node);
405 406 407 408 409 410
		kfree(root);
		return ERR_PTR(ret);
	}

	ret = btrfs_sysfs_add_root(root);
	if (ret) {
411
		free_extent_buffer(root->node);
412 413 414 415 416
		kfree(root->name);
		kfree(root);
		return ERR_PTR(ret);
	}

417 418 419 420
	ret = btrfs_find_dead_roots(fs_info->tree_root,
				    root->root_key.objectid, root);
	BUG_ON(ret);

421 422 423
	return root;
}

C
Chris Mason 已提交
424
struct btrfs_root *open_ctree(struct super_block *sb)
425
{
426 427 428 429
	u32 sectorsize;
	u32 nodesize;
	u32 leafsize;
	u32 blocksize;
C
Chris Mason 已提交
430 431 432 433 434 435
	struct btrfs_root *extent_root = kmalloc(sizeof(struct btrfs_root),
						 GFP_NOFS);
	struct btrfs_root *tree_root = kmalloc(sizeof(struct btrfs_root),
					       GFP_NOFS);
	struct btrfs_fs_info *fs_info = kmalloc(sizeof(*fs_info),
						GFP_NOFS);
436
	int ret;
C
Chris Mason 已提交
437
	int err = -EIO;
C
Chris Mason 已提交
438
	struct btrfs_super_block *disk_super;
439

C
Chris Mason 已提交
440 441 442 443
	if (!extent_root || !tree_root || !fs_info) {
		err = -ENOMEM;
		goto fail;
	}
444
	INIT_RADIX_TREE(&fs_info->fs_roots_radix, GFP_NOFS);
C
Chris Mason 已提交
445
	INIT_LIST_HEAD(&fs_info->trans_list);
446
	INIT_LIST_HEAD(&fs_info->dead_roots);
447 448
	memset(&fs_info->super_kobj, 0, sizeof(fs_info->super_kobj));
	init_completion(&fs_info->kobj_unregister);
C
Chris Mason 已提交
449
	sb_set_blocksize(sb, 4096);
450
	fs_info->running_transaction = NULL;
451
	fs_info->last_trans_committed = 0;
452 453
	fs_info->tree_root = tree_root;
	fs_info->extent_root = extent_root;
C
Chris Mason 已提交
454
	fs_info->sb = sb;
455 456
	fs_info->btree_inode = new_inode(sb);
	fs_info->btree_inode->i_ino = 1;
C
Chris Mason 已提交
457
	fs_info->btree_inode->i_nlink = 1;
458 459
	fs_info->btree_inode->i_size = sb->s_bdev->bd_inode->i_size;
	fs_info->btree_inode->i_mapping->a_ops = &btree_aops;
460 461 462
	extent_map_tree_init(&BTRFS_I(fs_info->btree_inode)->extent_tree,
			     fs_info->btree_inode->i_mapping,
			     GFP_NOFS);
463 464
	extent_map_tree_init(&fs_info->free_space_cache,
			     fs_info->btree_inode->i_mapping, GFP_NOFS);
465 466
	extent_map_tree_init(&fs_info->block_group_cache,
			     fs_info->btree_inode->i_mapping, GFP_NOFS);
467 468 469 470 471 472
	extent_map_tree_init(&fs_info->pinned_extents,
			     fs_info->btree_inode->i_mapping, GFP_NOFS);
	extent_map_tree_init(&fs_info->pending_del,
			     fs_info->btree_inode->i_mapping, GFP_NOFS);
	extent_map_tree_init(&fs_info->extent_ins,
			     fs_info->btree_inode->i_mapping, GFP_NOFS);
473
	fs_info->do_barriers = 1;
474 475
	fs_info->closing = 0;

C
Chris Mason 已提交
476
	INIT_DELAYED_WORK(&fs_info->trans_work, btrfs_transaction_cleaner);
477 478 479
	BTRFS_I(fs_info->btree_inode)->root = tree_root;
	memset(&BTRFS_I(fs_info->btree_inode)->location, 0,
	       sizeof(struct btrfs_key));
C
Chris Mason 已提交
480
	insert_inode_hash(fs_info->btree_inode);
481
	mapping_set_gfp_mask(fs_info->btree_inode->i_mapping, GFP_NOFS);
C
Chris Mason 已提交
482

C
Chris Mason 已提交
483
	mutex_init(&fs_info->trans_mutex);
C
Chris Mason 已提交
484
	mutex_init(&fs_info->fs_mutex);
485

486
	__setup_root(512, 512, 512, tree_root,
C
Chris Mason 已提交
487
		     fs_info, BTRFS_ROOT_TREE_OBJECTID);
488

C
Chris Mason 已提交
489
	fs_info->sb_buffer = read_tree_block(tree_root,
490 491
					     BTRFS_SUPER_INFO_OFFSET,
					     512);
492

493
	if (!fs_info->sb_buffer)
C
Chris Mason 已提交
494 495
		goto fail_iput;

496 497 498 499 500 501 502
	read_extent_buffer(fs_info->sb_buffer, &fs_info->super_copy, 0,
			   sizeof(fs_info->super_copy));

	read_extent_buffer(fs_info->sb_buffer, fs_info->fsid,
			   (unsigned long)btrfs_super_fsid(fs_info->sb_buffer),
			   BTRFS_FSID_SIZE);
	disk_super = &fs_info->super_copy;
503
	if (!btrfs_super_root(disk_super))
C
Chris Mason 已提交
504
		goto fail_sb_buffer;
505

506 507 508 509 510 511 512
	nodesize = btrfs_super_nodesize(disk_super);
	leafsize = btrfs_super_leafsize(disk_super);
	sectorsize = btrfs_super_sectorsize(disk_super);
	tree_root->nodesize = nodesize;
	tree_root->leafsize = leafsize;
	tree_root->sectorsize = sectorsize;

513
	i_size_write(fs_info->btree_inode,
514
		     btrfs_super_total_bytes(disk_super));
515

C
Chris Mason 已提交
516 517 518 519 520 521

	if (strncmp((char *)(&disk_super->magic), BTRFS_MAGIC,
		    sizeof(disk_super->magic))) {
		printk("btrfs: valid FS not found on %s\n", sb->s_id);
		goto fail_sb_buffer;
	}
522 523
	blocksize = btrfs_level_size(tree_root,
				     btrfs_super_root_level(disk_super));
C
Chris Mason 已提交
524
	tree_root->node = read_tree_block(tree_root,
525 526
					  btrfs_super_root(disk_super),
					  blocksize);
C
Chris Mason 已提交
527 528
	if (!tree_root->node)
		goto fail_sb_buffer;
529

530 531 532 533 534
#if 0
	btrfs_print_leaf(tree_root, tree_root->node);
	err = -EIO;
	goto fail_tree_root;
#endif
C
Chris Mason 已提交
535
	mutex_lock(&fs_info->fs_mutex);
536 537

	ret = find_and_setup_root(tree_root, fs_info,
C
Chris Mason 已提交
538
				  BTRFS_EXTENT_TREE_OBJECTID, extent_root);
C
Chris Mason 已提交
539 540 541 542
	if (ret) {
		mutex_unlock(&fs_info->fs_mutex);
		goto fail_tree_root;
	}
543

C
Chris Mason 已提交
544 545
	btrfs_read_block_groups(extent_root);

546
	fs_info->generation = btrfs_super_generation(disk_super) + 1;
C
Chris Mason 已提交
547
	mutex_unlock(&fs_info->fs_mutex);
548
	return tree_root;
C
Chris Mason 已提交
549 550

fail_tree_root:
551
	free_extent_buffer(tree_root->node);
C
Chris Mason 已提交
552
fail_sb_buffer:
553
	free_extent_buffer(fs_info->sb_buffer);
C
Chris Mason 已提交
554 555 556 557 558 559 560
fail_iput:
	iput(fs_info->btree_inode);
fail:
	kfree(extent_root);
	kfree(tree_root);
	kfree(fs_info);
	return ERR_PTR(err);
561 562
}

563
int write_ctree_super(struct btrfs_trans_handle *trans, struct btrfs_root
C
Chris Mason 已提交
564
		      *root)
565
{
566
	int ret;
567 568 569 570 571 572 573
	struct extent_buffer *super = root->fs_info->sb_buffer;
	struct inode *btree_inode = root->fs_info->btree_inode;

	set_extent_buffer_dirty(&BTRFS_I(btree_inode)->extent_tree, super);
	ret = sync_page_range_nolock(btree_inode, btree_inode->i_mapping,
				     super->start, super->len);
	return ret;
C
Chris Mason 已提交
574 575
}

576
int btrfs_free_fs_root(struct btrfs_fs_info *fs_info, struct btrfs_root *root)
C
Chris Mason 已提交
577 578 579
{
	radix_tree_delete(&fs_info->fs_roots_radix,
			  (unsigned long)root->root_key.objectid);
580
	btrfs_sysfs_del_root(root);
C
Chris Mason 已提交
581 582 583
	if (root->inode)
		iput(root->inode);
	if (root->node)
584
		free_extent_buffer(root->node);
C
Chris Mason 已提交
585
	if (root->commit_root)
586
		free_extent_buffer(root->commit_root);
587 588
	if (root->name)
		kfree(root->name);
C
Chris Mason 已提交
589 590 591 592
	kfree(root);
	return 0;
}

C
Chris Mason 已提交
593
static int del_fs_roots(struct btrfs_fs_info *fs_info)
594 595 596 597 598 599 600 601 602 603 604
{
	int ret;
	struct btrfs_root *gang[8];
	int i;

	while(1) {
		ret = radix_tree_gang_lookup(&fs_info->fs_roots_radix,
					     (void **)gang, 0,
					     ARRAY_SIZE(gang));
		if (!ret)
			break;
C
Chris Mason 已提交
605
		for (i = 0; i < ret; i++)
606
			btrfs_free_fs_root(fs_info, gang[i]);
607 608 609
	}
	return 0;
}
610

C
Chris Mason 已提交
611
int close_ctree(struct btrfs_root *root)
C
Chris Mason 已提交
612
{
613
	int ret;
614
	struct btrfs_trans_handle *trans;
615
	struct btrfs_fs_info *fs_info = root->fs_info;
616

617
	fs_info->closing = 1;
C
Chris Mason 已提交
618
	btrfs_transaction_flush_work(root);
619
	mutex_lock(&fs_info->fs_mutex);
620
	btrfs_defrag_dirty_roots(root->fs_info);
C
Chris Mason 已提交
621
	trans = btrfs_start_transaction(root, 1);
622
	ret = btrfs_commit_transaction(trans, root);
C
Chris Mason 已提交
623 624 625 626
	/* run commit again to  drop the original snapshot */
	trans = btrfs_start_transaction(root, 1);
	btrfs_commit_transaction(trans, root);
	ret = btrfs_write_and_wait_transaction(NULL, root);
627
	BUG_ON(ret);
C
Chris Mason 已提交
628
	write_ctree_super(NULL, root);
629 630 631
	mutex_unlock(&fs_info->fs_mutex);

	if (fs_info->extent_root->node)
632
		free_extent_buffer(fs_info->extent_root->node);
633

634
	if (fs_info->tree_root->node)
635
		free_extent_buffer(fs_info->tree_root->node);
636

637
	free_extent_buffer(fs_info->sb_buffer);
638

C
Chris Mason 已提交
639
	btrfs_free_block_groups(root->fs_info);
640
	del_fs_roots(fs_info);
641 642
	truncate_inode_pages(fs_info->btree_inode->i_mapping, 0);
	iput(fs_info->btree_inode);
643 644
	kfree(fs_info->extent_root);
	kfree(fs_info->tree_root);
645 646 647
	return 0;
}

648 649
int btrfs_buffer_uptodate(struct extent_buffer *buf)
{
650
	struct inode *btree_inode = buf->pages[0]->mapping->host;
651 652 653 654
	return extent_buffer_uptodate(&BTRFS_I(btree_inode)->extent_tree, buf);
}

int btrfs_set_buffer_uptodate(struct extent_buffer *buf)
C
Chris Mason 已提交
655
{
656
	struct inode *btree_inode = buf->pages[0]->mapping->host;
657 658 659
	return set_extent_buffer_uptodate(&BTRFS_I(btree_inode)->extent_tree,
					  buf);
}
660

661 662
void btrfs_mark_buffer_dirty(struct extent_buffer *buf)
{
663
	struct btrfs_root *root = BTRFS_I(buf->pages[0]->mapping->host)->root;
664 665
	u64 transid = btrfs_header_generation(buf);
	struct inode *btree_inode = root->fs_info->btree_inode;
666

C
Chris Mason 已提交
667 668
	if (transid != root->fs_info->generation) {
		printk(KERN_CRIT "transid mismatch buffer %llu, found %Lu running %Lu\n",
669
			(unsigned long long)buf->start,
C
Chris Mason 已提交
670 671 672
			transid, root->fs_info->generation);
		WARN_ON(1);
	}
673
	set_extent_buffer_dirty(&BTRFS_I(btree_inode)->extent_tree, buf);
674 675
}

676
void btrfs_btree_balance_dirty(struct btrfs_root *root, unsigned long nr)
C
Chris Mason 已提交
677
{
678 679
	balance_dirty_pages_ratelimited_nr(
			root->fs_info->btree_inode->i_mapping, nr);
C
Chris Mason 已提交
680
}