dm btree: add a defensive bounds check to insert_at()
authorJoe Thornber <ejt@redhat.com>
Fri, 10 Dec 2021 13:44:13 +0000 (13:44 +0000)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Thu, 27 Jan 2022 07:47:38 +0000 (08:47 +0100)
[ Upstream commit 85bca3c05b6cca31625437eedf2060e846c4bbad ]

Corrupt metadata could trigger an out of bounds write.

Signed-off-by: Joe Thornber <ejt@redhat.com>
Signed-off-by: Mike Snitzer <snitzer@redhat.com>
Signed-off-by: Sasha Levin <sashal@kernel.org>
drivers/md/persistent-data/dm-btree.c

index 386215245dfe27d0df9b1b30fb0482218df5c138..85273da5da2066281b3dd48b3b1419247fe7d918 100644 (file)
@@ -83,14 +83,16 @@ void inc_children(struct dm_transaction_manager *tm, struct btree_node *n,
 }
 
 static int insert_at(size_t value_size, struct btree_node *node, unsigned index,
-                     uint64_t key, void *value)
-                     __dm_written_to_disk(value)
+                    uint64_t key, void *value)
+       __dm_written_to_disk(value)
 {
        uint32_t nr_entries = le32_to_cpu(node->header.nr_entries);
+       uint32_t max_entries = le32_to_cpu(node->header.max_entries);
        __le64 key_le = cpu_to_le64(key);
 
        if (index > nr_entries ||
-           index >= le32_to_cpu(node->header.max_entries)) {
+           index >= max_entries ||
+           nr_entries >= max_entries) {
                DMERR("too many entries in btree node for insert");
                __dm_unbless_for_disk(value);
                return -ENOMEM;