Received: by 2002:ad5:474a:0:0:0:0:0 with SMTP id i10csp10791826imu; Thu, 6 Dec 2018 06:54:12 -0800 (PST) X-Google-Smtp-Source: AFSGD/Vh4xigAG1R7KcZRT/pTpyTAIrWlE9S45i9A56a18/AKBcswyR9lwvghdlSt2hrJeogDXoe X-Received: by 2002:a65:560e:: with SMTP id l14mr24407045pgs.168.1544108052496; Thu, 06 Dec 2018 06:54:12 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1544108052; cv=none; d=google.com; s=arc-20160816; b=RPMlmreJetzrW8ZEejFsfuK2QJjlp1lgTau08QL6kA9UfuAaTTP3rK4jN7f/ygNKWN Gakzo9B0WInFHK6qG830AKlhDF4sc/mHJ4eQQ95cbfwaiEXnikA9quiXYDFmhpCoS8bt isrcx2Aj9C8xjYg6t3J7dWNc4hxIInOfo7xxHPkh6V8EDUwxg15sFhAw6XGrjz0dHSfy TEvalrqGSaww6wBXzojH1ByU6/pY+qbbgGQGjK1Db/5yM4ty2WNQXtb4AiGkTqAsIC66 I7Jh4fBoizCIOiulrpwAYfWMv4/oiOCBylKDaQGGvCA7aogq0Qnsio+h2CFHvVbopCP5 /CSw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :user-agent:references:in-reply-to:message-id:date:subject:cc:to :from:dkim-signature; bh=Or/zJhkdU4kDh+5R6NNkvILu/2ppYvg+FZK18M2aH7I=; b=nr9oMkWc2J8vtqdnrq6O6jHwF2ygT768v8DK36U+kjbuosTeDwtkXBbAzJOy6yb1BA rSLmIblowFTGhTpAm26EnfCaMeh677Ml5GlNk0MvSLbZvGttteDoad+hMApFNqt0OfGN NV+q39a65cm1P3I0LCKh7V8uGIGoeU+0vWM3+owOdoF1tOYQHuIf4WF+C4VHplYEIS82 0N3kF8rjHFqlL+Ncr+Lq0i1q4Hpnt+t4JYIWYwhR/vZl0Nkbwkk+xMtm74SG2oAldaGS NEPCN+0R5+n2rurKQfpQ85CbaTHaC8c4aM5zi7mMEjOPZCSpY2TLMVOkIpQqG0Dx0Fja bE8w== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b=2t6IyX4N; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id l61si449219plb.6.2018.12.06.06.53.56; Thu, 06 Dec 2018 06:54:12 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b=2t6IyX4N; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1726017AbeLFOvi (ORCPT + 99 others); Thu, 6 Dec 2018 09:51:38 -0500 Received: from mail.kernel.org ([198.145.29.99]:51476 "EHLO mail.kernel.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1731297AbeLFOqn (ORCPT ); Thu, 6 Dec 2018 09:46:43 -0500 Received: from localhost (5356596B.cm-6-7b.dynamic.ziggo.nl [83.86.89.107]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by mail.kernel.org (Postfix) with ESMTPSA id 0E7C721479; Thu, 6 Dec 2018 14:46:41 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=kernel.org; s=default; t=1544107602; bh=YKp3jwXFsI6miLoa7chgiyQV9YDVwT9jwd/jlZT5bac=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=2t6IyX4NkCkqzmzHezAbDS7G7ydJPwBHTmSv9YqnMYorUSTnti0ren4uU/dgz3LnZ 4LoNAtszkTWWLsP8miHNyVQ5gL3+GG3Pwxu9jXo55m0ZC1+ENPg6Zr6a4p0kCktAJj XdMsxhcF1rbaJNUluOopD+U4NAu0UuQZDc78zh5o= From: Greg Kroah-Hartman To: linux-kernel@vger.kernel.org Cc: Greg Kroah-Hartman , stable@vger.kernel.org, Qu Wenruo , Nikolay Borisov , David Sterba , Ben Hutchings Subject: [PATCH 4.9 067/101] btrfs: Refactor check_leaf function for later expansion Date: Thu, 6 Dec 2018 15:39:06 +0100 Message-Id: <20181206143015.611313000@linuxfoundation.org> X-Mailer: git-send-email 2.19.2 In-Reply-To: <20181206143011.174892052@linuxfoundation.org> References: <20181206143011.174892052@linuxfoundation.org> User-Agent: quilt/0.65 X-stable: review X-Patchwork-Hint: ignore MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org 4.9-stable review patch. If anyone has any objections, please let me know. ------------------ From: Qu Wenruo commit c3267bbaa9cae09b62960eafe33ad19196803285 upstream. Current check_leaf() function does a good job checking key order and item offset/size. However it only checks from slot 0 to the last but one slot, this is good but makes later expansion hard. So this refactoring iterates from slot 0 to the last slot. For key comparison, it uses a key with all 0 as initial key, so all valid keys should be larger than that. And for item size/offset checks, it compares current item end with previous item offset. For slot 0, use leaf end as a special case. This makes later item/key offset checks and item size checks easier to be implemented. Also, makes check_leaf() to return -EUCLEAN other than -EIO to indicate error. Signed-off-by: Qu Wenruo Reviewed-by: Nikolay Borisov Reviewed-by: David Sterba Signed-off-by: David Sterba [bwh: Backported to 4.9: - BTRFS_LEAF_DATA_SIZE() takes a root rather than an fs_info - Adjust context] Signed-off-by: Ben Hutchings Signed-off-by: Greg Kroah-Hartman --- fs/btrfs/disk-io.c | 50 +++++++++++++++++++++++++++----------------------- 1 file changed, 27 insertions(+), 23 deletions(-) --- a/fs/btrfs/disk-io.c +++ b/fs/btrfs/disk-io.c @@ -547,8 +547,9 @@ static int check_tree_block_fsid(struct static noinline int check_leaf(struct btrfs_root *root, struct extent_buffer *leaf) { + /* No valid key type is 0, so all key should be larger than this key */ + struct btrfs_key prev_key = {0, 0, 0}; struct btrfs_key key; - struct btrfs_key leaf_key; u32 nritems = btrfs_header_nritems(leaf); int slot; @@ -581,7 +582,7 @@ static noinline int check_leaf(struct bt CORRUPT("non-root leaf's nritems is 0", leaf, check_root, 0); free_extent_buffer(eb); - return -EIO; + return -EUCLEAN; } free_extent_buffer(eb); } @@ -591,28 +592,23 @@ static noinline int check_leaf(struct bt if (nritems == 0) return 0; - /* Check the 0 item */ - if (btrfs_item_offset_nr(leaf, 0) + btrfs_item_size_nr(leaf, 0) != - BTRFS_LEAF_DATA_SIZE(root)) { - CORRUPT("invalid item offset size pair", leaf, root, 0); - return -EIO; - } - /* - * Check to make sure each items keys are in the correct order and their - * offsets make sense. We only have to loop through nritems-1 because - * we check the current slot against the next slot, which verifies the - * next slot's offset+size makes sense and that the current's slot - * offset is correct. + * Check the following things to make sure this is a good leaf, and + * leaf users won't need to bother with similar sanity checks: + * + * 1) key order + * 2) item offset and size + * No overlap, no hole, all inside the leaf. */ - for (slot = 0; slot < nritems - 1; slot++) { - btrfs_item_key_to_cpu(leaf, &leaf_key, slot); - btrfs_item_key_to_cpu(leaf, &key, slot + 1); + for (slot = 0; slot < nritems; slot++) { + u32 item_end_expected; + + btrfs_item_key_to_cpu(leaf, &key, slot); /* Make sure the keys are in the right order */ - if (btrfs_comp_cpu_keys(&leaf_key, &key) >= 0) { + if (btrfs_comp_cpu_keys(&prev_key, &key) >= 0) { CORRUPT("bad key order", leaf, root, slot); - return -EIO; + return -EUCLEAN; } /* @@ -620,10 +616,14 @@ static noinline int check_leaf(struct bt * item data starts at the end of the leaf and grows towards the * front. */ - if (btrfs_item_offset_nr(leaf, slot) != - btrfs_item_end_nr(leaf, slot + 1)) { + if (slot == 0) + item_end_expected = BTRFS_LEAF_DATA_SIZE(root); + else + item_end_expected = btrfs_item_offset_nr(leaf, + slot - 1); + if (btrfs_item_end_nr(leaf, slot) != item_end_expected) { CORRUPT("slot offset bad", leaf, root, slot); - return -EIO; + return -EUCLEAN; } /* @@ -634,8 +634,12 @@ static noinline int check_leaf(struct bt if (btrfs_item_end_nr(leaf, slot) > BTRFS_LEAF_DATA_SIZE(root)) { CORRUPT("slot end outside of leaf", leaf, root, slot); - return -EIO; + return -EUCLEAN; } + + prev_key.objectid = key.objectid; + prev_key.type = key.type; + prev_key.offset = key.offset; } return 0;