root-tree.c 13.5 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.
 */

19
#include <linux/uuid.h>
20
#include "ctree.h"
21
#include "transaction.h"
22 23 24
#include "disk-io.h"
#include "print-tree.h"

25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66
/*
 * Read a root item from the tree. In case we detect a root item smaller then
 * sizeof(root_item), we know it's an old version of the root structure and
 * initialize all new fields to zero. The same happens if we detect mismatching
 * generation numbers as then we know the root was once mounted with an older
 * kernel that was not aware of the root item structure change.
 */
void btrfs_read_root_item(struct btrfs_root *root,
			 struct extent_buffer *eb, int slot,
			 struct btrfs_root_item *item)
{
	uuid_le uuid;
	int len;
	int need_reset = 0;

	len = btrfs_item_size_nr(eb, slot);
	read_extent_buffer(eb, item, btrfs_item_ptr_offset(eb, slot),
			min_t(int, len, (int)sizeof(*item)));
	if (len < sizeof(*item))
		need_reset = 1;
	if (!need_reset && btrfs_root_generation(item)
		!= btrfs_root_generation_v2(item)) {
		if (btrfs_root_generation_v2(item) != 0) {
			printk(KERN_WARNING "btrfs: mismatching "
					"generation and generation_v2 "
					"found in root item. This root "
					"was probably mounted with an "
					"older kernel. Resetting all "
					"new fields.\n");
		}
		need_reset = 1;
	}
	if (need_reset) {
		memset(&item->generation_v2, 0,
			sizeof(*item) - offsetof(struct btrfs_root_item,
					generation_v2));

		uuid_le_gen(&uuid);
		memcpy(item->uuid, uuid.b, BTRFS_UUID_SIZE);
	}
}

C
Chris Mason 已提交
67 68 69 70 71
/*
 * lookup the root with the highest offset for a given objectid.  The key we do
 * find is copied into 'key'.  If we find something return 0, otherwise 1, < 0
 * on error.
 */
72 73 74
int btrfs_find_last_root(struct btrfs_root *root, u64 objectid,
			struct btrfs_root_item *item, struct btrfs_key *key)
{
75
	struct btrfs_path *path;
76
	struct btrfs_key search_key;
77 78
	struct btrfs_key found_key;
	struct extent_buffer *l;
79 80 81 82
	int ret;
	int slot;

	search_key.objectid = objectid;
83
	search_key.type = BTRFS_ROOT_ITEM_KEY;
84
	search_key.offset = (u64)-1;
85

86
	path = btrfs_alloc_path();
T
Tsutomu Itoh 已提交
87 88
	if (!path)
		return -ENOMEM;
89
	ret = btrfs_search_slot(NULL, root, &search_key, path, 0, 0);
90 91
	if (ret < 0)
		goto out;
92

93
	BUG_ON(ret == 0);
94 95 96 97
	if (path->slots[0] == 0) {
		ret = 1;
		goto out;
	}
98
	l = path->nodes[0];
99
	slot = path->slots[0] - 1;
100
	btrfs_item_key_to_cpu(l, &found_key, slot);
101 102
	if (found_key.objectid != objectid ||
	    found_key.type != BTRFS_ROOT_ITEM_KEY) {
103 104 105
		ret = 1;
		goto out;
	}
106
	if (item)
107
		btrfs_read_root_item(root, l, slot, item);
108 109
	if (key)
		memcpy(key, &found_key, sizeof(found_key));
110

111 112
	ret = 0;
out:
113
	btrfs_free_path(path);
114 115 116
	return ret;
}

117 118
void btrfs_set_root_node(struct btrfs_root_item *item,
			 struct extent_buffer *node)
119 120 121 122 123 124
{
	btrfs_set_root_bytenr(item, node->start);
	btrfs_set_root_level(item, btrfs_header_level(node));
	btrfs_set_root_generation(item, btrfs_header_generation(node));
}

C
Chris Mason 已提交
125 126 127
/*
 * copy the data in 'item' into the btree
 */
128 129 130
int btrfs_update_root(struct btrfs_trans_handle *trans, struct btrfs_root
		      *root, struct btrfs_key *key, struct btrfs_root_item
		      *item)
131
{
132
	struct btrfs_path *path;
133
	struct extent_buffer *l;
134 135
	int ret;
	int slot;
136
	unsigned long ptr;
137
	int old_len;
138

139
	path = btrfs_alloc_path();
140 141 142
	if (!path)
		return -ENOMEM;

143
	ret = btrfs_search_slot(trans, root, key, path, 0, 1);
144 145
	if (ret < 0)
		goto out_abort;
C
Chris Mason 已提交
146 147 148

	if (ret != 0) {
		btrfs_print_leaf(root, path->nodes[0]);
C
Chris Mason 已提交
149 150 151
		printk(KERN_CRIT "unable to update root key %llu %u %llu\n",
		       (unsigned long long)key->objectid, key->type,
		       (unsigned long long)key->offset);
C
Chris Mason 已提交
152 153 154
		BUG_ON(1);
	}

155
	l = path->nodes[0];
156
	slot = path->slots[0];
157
	ptr = btrfs_item_ptr_offset(l, slot);
158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189
	old_len = btrfs_item_size_nr(l, slot);

	/*
	 * If this is the first time we update the root item which originated
	 * from an older kernel, we need to enlarge the item size to make room
	 * for the added fields.
	 */
	if (old_len < sizeof(*item)) {
		btrfs_release_path(path);
		ret = btrfs_search_slot(trans, root, key, path,
				-1, 1);
		if (ret < 0)
			goto out_abort;
		ret = btrfs_del_item(trans, root, path);
		if (ret < 0)
			goto out_abort;
		btrfs_release_path(path);
		ret = btrfs_insert_empty_item(trans, root, path,
				key, sizeof(*item));
		if (ret < 0)
			goto out_abort;
		l = path->nodes[0];
		slot = path->slots[0];
		ptr = btrfs_item_ptr_offset(l, slot);
	}

	/*
	 * Update generation_v2 so at the next mount we know the new root
	 * fields are valid.
	 */
	btrfs_set_root_generation_v2(item, btrfs_root_generation(item));

190
	write_extent_buffer(l, item, ptr, sizeof(*item));
191
	btrfs_mark_buffer_dirty(path->nodes[0]);
192
out:
193
	btrfs_free_path(path);
194
	return ret;
195 196 197 198

out_abort:
	btrfs_abort_transaction(trans, root, ret);
	goto out;
199 200
}

J
Jeff Mahoney 已提交
201 202
int btrfs_insert_root(struct btrfs_trans_handle *trans, struct btrfs_root *root,
		      struct btrfs_key *key, struct btrfs_root_item *item)
203
{
204 205 206 207
	/*
	 * Make sure generation v1 and v2 match. See update_root for details.
	 */
	btrfs_set_root_generation_v2(item, btrfs_root_generation(item));
J
Jeff Mahoney 已提交
208
	return btrfs_insert_item(trans, root, key, item, sizeof(*item));
209 210
}

C
Chris Mason 已提交
211 212
/*
 * at mount time we want to find all the old transaction snapshots that were in
C
Chris Mason 已提交
213 214 215
 * the process of being deleted if we crashed.  This is any root item with an
 * offset lower than the latest root.  They need to be queued for deletion to
 * finish what was happening when we crashed.
C
Chris Mason 已提交
216
 */
217
int btrfs_find_dead_roots(struct btrfs_root *root, u64 objectid)
218 219 220 221
{
	struct btrfs_root *dead_root;
	struct btrfs_root_item *ri;
	struct btrfs_key key;
222
	struct btrfs_key found_key;
223 224 225
	struct btrfs_path *path;
	int ret;
	u32 nritems;
226
	struct extent_buffer *leaf;
227 228
	int slot;

229
	key.objectid = objectid;
230 231 232 233 234
	btrfs_set_key_type(&key, BTRFS_ROOT_ITEM_KEY);
	key.offset = 0;
	path = btrfs_alloc_path();
	if (!path)
		return -ENOMEM;
235 236

again:
237 238 239
	ret = btrfs_search_slot(NULL, root, &key, path, 0, 0);
	if (ret < 0)
		goto err;
C
Chris Mason 已提交
240
	while (1) {
241 242
		leaf = path->nodes[0];
		nritems = btrfs_header_nritems(leaf);
243 244 245 246 247
		slot = path->slots[0];
		if (slot >= nritems) {
			ret = btrfs_next_leaf(root, path);
			if (ret)
				break;
248 249
			leaf = path->nodes[0];
			nritems = btrfs_header_nritems(leaf);
250 251
			slot = path->slots[0];
		}
252
		btrfs_item_key_to_cpu(leaf, &key, slot);
253 254
		if (btrfs_key_type(&key) != BTRFS_ROOT_ITEM_KEY)
			goto next;
255 256 257 258 259 260 261

		if (key.objectid < objectid)
			goto next;

		if (key.objectid > objectid)
			break;

262
		ri = btrfs_item_ptr(leaf, slot, struct btrfs_root_item);
263
		if (btrfs_disk_root_refs(leaf, ri) != 0)
264
			goto next;
265

266 267
		memcpy(&found_key, &key, sizeof(key));
		key.offset++;
268
		btrfs_release_path(path);
269 270 271
		dead_root =
			btrfs_read_fs_root_no_radix(root->fs_info->tree_root,
						    &found_key);
A
Aneesh 已提交
272 273
		if (IS_ERR(dead_root)) {
			ret = PTR_ERR(dead_root);
274 275
			goto err;
		}
276

277
		ret = btrfs_add_dead_root(dead_root);
278 279
		if (ret)
			goto err;
280
		goto again;
281 282 283 284 285 286 287 288 289 290
next:
		slot++;
		path->slots[0]++;
	}
	ret = 0;
err:
	btrfs_free_path(path);
	return ret;
}

291 292 293 294 295
int btrfs_find_orphan_roots(struct btrfs_root *tree_root)
{
	struct extent_buffer *leaf;
	struct btrfs_path *path;
	struct btrfs_key key;
296 297
	struct btrfs_key root_key;
	struct btrfs_root *root;
298 299 300 301 302 303 304 305 306 307 308
	int err = 0;
	int ret;

	path = btrfs_alloc_path();
	if (!path)
		return -ENOMEM;

	key.objectid = BTRFS_ORPHAN_OBJECTID;
	key.type = BTRFS_ORPHAN_ITEM_KEY;
	key.offset = 0;

309 310 311
	root_key.type = BTRFS_ROOT_ITEM_KEY;
	root_key.offset = (u64)-1;

312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329
	while (1) {
		ret = btrfs_search_slot(NULL, tree_root, &key, path, 0, 0);
		if (ret < 0) {
			err = ret;
			break;
		}

		leaf = path->nodes[0];
		if (path->slots[0] >= btrfs_header_nritems(leaf)) {
			ret = btrfs_next_leaf(tree_root, path);
			if (ret < 0)
				err = ret;
			if (ret != 0)
				break;
			leaf = path->nodes[0];
		}

		btrfs_item_key_to_cpu(leaf, &key, path->slots[0]);
330
		btrfs_release_path(path);
331 332 333 334 335

		if (key.objectid != BTRFS_ORPHAN_OBJECTID ||
		    key.type != BTRFS_ORPHAN_ITEM_KEY)
			break;

336 337 338 339 340 341 342 343 344 345
		root_key.objectid = key.offset;
		key.offset++;

		root = btrfs_read_fs_root_no_name(tree_root->fs_info,
						  &root_key);
		if (!IS_ERR(root))
			continue;

		ret = PTR_ERR(root);
		if (ret != -ENOENT) {
346 347 348 349
			err = ret;
			break;
		}

350 351 352 353 354
		ret = btrfs_find_dead_roots(tree_root, root_key.objectid);
		if (ret) {
			err = ret;
			break;
		}
355 356 357 358 359 360
	}

	btrfs_free_path(path);
	return err;
}

C
Chris Mason 已提交
361
/* drop the root item for 'key' from 'root' */
362 363
int btrfs_del_root(struct btrfs_trans_handle *trans, struct btrfs_root *root,
		   struct btrfs_key *key)
364
{
365
	struct btrfs_path *path;
366
	int ret;
C
Chris Mason 已提交
367
	struct btrfs_root_item *ri;
368
	struct extent_buffer *leaf;
369

370
	path = btrfs_alloc_path();
T
Tsutomu Itoh 已提交
371 372
	if (!path)
		return -ENOMEM;
373
	ret = btrfs_search_slot(trans, root, key, path, -1, 1);
374 375
	if (ret < 0)
		goto out;
376

377
	BUG_ON(ret != 0);
378 379
	leaf = path->nodes[0];
	ri = btrfs_item_ptr(leaf, path->slots[0], struct btrfs_root_item);
C
Chris Mason 已提交
380

381
	ret = btrfs_del_item(trans, root, path);
382
out:
383
	btrfs_free_path(path);
384 385
	return ret;
}
386 387 388

int btrfs_del_root_ref(struct btrfs_trans_handle *trans,
		       struct btrfs_root *tree_root,
389 390 391
		       u64 root_id, u64 ref_id, u64 dirid, u64 *sequence,
		       const char *name, int name_len)

392
{
393 394 395
	struct btrfs_path *path;
	struct btrfs_root_ref *ref;
	struct extent_buffer *leaf;
396
	struct btrfs_key key;
397 398
	unsigned long ptr;
	int err = 0;
399 400 401
	int ret;

	path = btrfs_alloc_path();
402 403
	if (!path)
		return -ENOMEM;
404 405

	key.objectid = root_id;
406
	key.type = BTRFS_ROOT_BACKREF_KEY;
407
	key.offset = ref_id;
408
again:
409
	ret = btrfs_search_slot(trans, tree_root, &key, path, -1, 1);
410 411 412 413 414 415 416 417 418 419 420 421 422
	BUG_ON(ret < 0);
	if (ret == 0) {
		leaf = path->nodes[0];
		ref = btrfs_item_ptr(leaf, path->slots[0],
				     struct btrfs_root_ref);

		WARN_ON(btrfs_root_ref_dirid(leaf, ref) != dirid);
		WARN_ON(btrfs_root_ref_name_len(leaf, ref) != name_len);
		ptr = (unsigned long)(ref + 1);
		WARN_ON(memcmp_extent_buffer(leaf, name, ptr, name_len));
		*sequence = btrfs_root_ref_sequence(leaf, ref);

		ret = btrfs_del_item(trans, tree_root, path);
423 424 425 426
		if (ret) {
			err = ret;
			goto out;
		}
427 428 429 430
	} else
		err = -ENOENT;

	if (key.type == BTRFS_ROOT_BACKREF_KEY) {
431
		btrfs_release_path(path);
432 433 434 435 436
		key.objectid = ref_id;
		key.type = BTRFS_ROOT_REF_KEY;
		key.offset = root_id;
		goto again;
	}
437

438
out:
439
	btrfs_free_path(path);
440
	return err;
441 442
}

443 444 445 446 447 448 449 450 451 452 453 454 455 456 457
int btrfs_find_root_ref(struct btrfs_root *tree_root,
		   struct btrfs_path *path,
		   u64 root_id, u64 ref_id)
{
	struct btrfs_key key;
	int ret;

	key.objectid = root_id;
	key.type = BTRFS_ROOT_REF_KEY;
	key.offset = ref_id;

	ret = btrfs_search_slot(NULL, tree_root, &key, path, 0, 0);
	return ret;
}

458 459 460 461 462 463 464 465 466 467 468 469
/*
 * add a btrfs_root_ref item.  type is either BTRFS_ROOT_REF_KEY
 * or BTRFS_ROOT_BACKREF_KEY.
 *
 * The dirid, sequence, name and name_len refer to the directory entry
 * that is referencing the root.
 *
 * For a forward ref, the root_id is the id of the tree referencing
 * the root and ref_id is the id of the subvol  or snapshot.
 *
 * For a back ref the root_id is the id of the subvol or snapshot and
 * ref_id is the id of the tree referencing it.
470 471
 *
 * Will return 0, -ENOMEM, or anything from the CoW path
472 473 474
 */
int btrfs_add_root_ref(struct btrfs_trans_handle *trans,
		       struct btrfs_root *tree_root,
475
		       u64 root_id, u64 ref_id, u64 dirid, u64 sequence,
476 477 478 479 480 481 482 483 484 485
		       const char *name, int name_len)
{
	struct btrfs_key key;
	int ret;
	struct btrfs_path *path;
	struct btrfs_root_ref *ref;
	struct extent_buffer *leaf;
	unsigned long ptr;

	path = btrfs_alloc_path();
486 487
	if (!path)
		return -ENOMEM;
488 489

	key.objectid = root_id;
490
	key.type = BTRFS_ROOT_BACKREF_KEY;
491
	key.offset = ref_id;
492
again:
493 494
	ret = btrfs_insert_empty_item(trans, tree_root, path, &key,
				      sizeof(*ref) + name_len);
495 496 497 498 499
	if (ret) {
		btrfs_abort_transaction(trans, tree_root, ret);
		btrfs_free_path(path);
		return ret;
	}
500 501 502 503 504 505 506 507 508 509

	leaf = path->nodes[0];
	ref = btrfs_item_ptr(leaf, path->slots[0], struct btrfs_root_ref);
	btrfs_set_root_ref_dirid(leaf, ref, dirid);
	btrfs_set_root_ref_sequence(leaf, ref, sequence);
	btrfs_set_root_ref_name_len(leaf, ref, name_len);
	ptr = (unsigned long)(ref + 1);
	write_extent_buffer(leaf, name, ptr, name_len);
	btrfs_mark_buffer_dirty(leaf);

510
	if (key.type == BTRFS_ROOT_BACKREF_KEY) {
511
		btrfs_release_path(path);
512 513 514 515 516 517
		key.objectid = ref_id;
		key.type = BTRFS_ROOT_REF_KEY;
		key.offset = root_id;
		goto again;
	}

518
	btrfs_free_path(path);
519
	return 0;
520
}
521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538

/*
 * Old btrfs forgets to init root_item->flags and root_item->byte_limit
 * for subvolumes. To work around this problem, we steal a bit from
 * root_item->inode_item->flags, and use it to indicate if those fields
 * have been properly initialized.
 */
void btrfs_check_and_init_root_item(struct btrfs_root_item *root_item)
{
	u64 inode_flags = le64_to_cpu(root_item->inode.flags);

	if (!(inode_flags & BTRFS_INODE_ROOT_ITEM_INIT)) {
		inode_flags |= BTRFS_INODE_ROOT_ITEM_INIT;
		root_item->inode.flags = cpu_to_le64(inode_flags);
		root_item->flags = 0;
		root_item->byte_limit = 0;
	}
}
539 540 541 542 543 544 545 546

void btrfs_update_root_times(struct btrfs_trans_handle *trans,
			     struct btrfs_root *root)
{
	struct btrfs_root_item *item = &root->root_item;
	struct timespec ct = CURRENT_TIME;

	spin_lock(&root->root_times_lock);
547
	item->ctransid = cpu_to_le64(trans->transid);
548
	item->ctime.sec = cpu_to_le64(ct.tv_sec);
549
	item->ctime.nsec = cpu_to_le32(ct.tv_nsec);
550 551
	spin_unlock(&root->root_times_lock);
}