reada_for_search(fs_info, p, parent_level, slot, key->objectid);
/* first we do an atomic uptodate check */
- if (btrfs_buffer_uptodate(tmp, check.transid, true) > 0) {
+ if (btrfs_buffer_uptodate(tmp, check.transid, true, NULL) > 0) {
/*
* Do extra check for first_key, eb can be stale due to
* being cached, read from scrub, or have multiple
* detect blocks that either didn't get written at all or got written
* in the wrong place.
*/
-int btrfs_buffer_uptodate(struct extent_buffer *eb, u64 parent_transid, bool atomic)
+int btrfs_buffer_uptodate(struct extent_buffer *eb, u64 parent_transid, bool atomic,
+ const struct btrfs_tree_parent_check *check)
{
if (!extent_buffer_uptodate(eb))
return 0;
- if (!parent_transid || btrfs_header_generation(eb) == parent_transid)
+ if (!parent_transid || btrfs_header_generation(eb) == parent_transid) {
+ /*
+ * On a cache hit, the caller may still need tree parent
+ * verification before reusing the buffer.
+ */
+ if (unlikely(check && btrfs_verify_level_key(eb, check)))
+ return -EUCLEAN;
return 1;
+ }
if (atomic)
return -EAGAIN;
root->node = NULL;
goto fail;
}
- if (unlikely(!btrfs_buffer_uptodate(root->node, generation, false))) {
- ret = -EIO;
+
+ ret = btrfs_buffer_uptodate(root->node, generation, false, &check);
+ if (unlikely(ret <= 0)) {
+ if (ret == 0)
+ ret = -EIO;
goto fail;
}
void btrfs_put_root(struct btrfs_root *root);
void btrfs_mark_buffer_dirty(struct btrfs_trans_handle *trans,
struct extent_buffer *buf);
-int btrfs_buffer_uptodate(struct extent_buffer *buf, u64 parent_transid, bool atomic);
+int btrfs_buffer_uptodate(struct extent_buffer *buf, u64 parent_transid, bool atomic,
+ const struct btrfs_tree_parent_check *check);
int btrfs_read_extent_buffer(struct extent_buffer *buf,
const struct btrfs_tree_parent_check *check);
generation = btrfs_node_ptr_generation(path->nodes[level], path->slots[level]);
- if (btrfs_buffer_uptodate(next, generation, false))
+ if (btrfs_buffer_uptodate(next, generation, false, NULL))
return 0;
check.level = level - 1;
struct btrfs_fs_info *fs_info = eb->fs_info;
struct btrfs_bio *bbio;
- if (extent_buffer_uptodate(eb))
+ if (extent_buffer_uptodate(eb)) {
+ int ret;
+
+ ret = btrfs_buffer_uptodate(eb, 0, true, check);
+ if (unlikely(ret <= 0)) {
+ if (ret == 0)
+ ret = -EIO;
+ return ret;
+ }
return 0;
+ }
/*
* We could have had EXTENT_BUFFER_UPTODATE cleared by the write
* will now be set, and we shouldn't read it in again.
*/
if (unlikely(extent_buffer_uptodate(eb))) {
+ int ret;
+
clear_extent_buffer_reading(eb);
+ ret = btrfs_buffer_uptodate(eb, 0, true, check);
+ if (unlikely(ret <= 0)) {
+ if (ret == 0)
+ ret = -EIO;
+ return ret;
+ }
return 0;
}
if (IS_ERR(eb))
return;
- if (btrfs_buffer_uptodate(eb, gen, true)) {
+ if (btrfs_buffer_uptodate(eb, gen, true, NULL)) {
free_extent_buffer(eb);
return;
}
return ret;
}
- if (btrfs_buffer_uptodate(eb, gen, false) && level == 0) {
+ if (btrfs_buffer_uptodate(eb, gen, false, NULL) && level == 0) {
ret = btrfs_exclude_logged_extents(eb);
if (ret)
btrfs_abort_transaction(trans, ret);