| // SPDX-License-Identifier: GPL-2.0-only |
| |
| #include <linux/list.h> |
| #include <linux/kernel.h> |
| #include <linux/dm-verity-loadpin.h> |
| |
| #include "dm.h" |
| #include "dm-core.h" |
| #include "dm-verity.h" |
| |
| #define DM_MSG_PREFIX "verity-loadpin" |
| |
| LIST_HEAD(dm_verity_loadpin_trusted_root_digests); |
| |
| static bool is_trusted_verity_target(struct dm_target *ti) |
| { |
| int verity_mode; |
| u8 *root_digest; |
| unsigned int digest_size; |
| struct dm_verity_loadpin_trusted_root_digest *trd; |
| bool trusted = false; |
| |
| if (!dm_is_verity_target(ti)) |
| return false; |
| |
| verity_mode = dm_verity_get_mode(ti); |
| |
| if ((verity_mode != DM_VERITY_MODE_EIO) && |
| (verity_mode != DM_VERITY_MODE_RESTART) && |
| (verity_mode != DM_VERITY_MODE_PANIC)) |
| return false; |
| |
| if (dm_verity_get_root_digest(ti, &root_digest, &digest_size)) |
| return false; |
| |
| list_for_each_entry(trd, &dm_verity_loadpin_trusted_root_digests, node) { |
| if ((trd->len == digest_size) && |
| !memcmp(trd->data, root_digest, digest_size)) { |
| trusted = true; |
| break; |
| } |
| } |
| |
| kfree(root_digest); |
| |
| return trusted; |
| } |
| |
| /* |
| * Determines whether the file system of a superblock is located on |
| * a verity device that is trusted by LoadPin. |
| */ |
| bool dm_verity_loadpin_is_bdev_trusted(struct block_device *bdev) |
| { |
| struct mapped_device *md; |
| struct dm_table *table; |
| struct dm_target *ti; |
| int srcu_idx; |
| bool trusted = false; |
| |
| if (bdev == NULL) |
| return false; |
| |
| if (list_empty(&dm_verity_loadpin_trusted_root_digests)) |
| return false; |
| |
| md = dm_get_md(bdev->bd_dev); |
| if (!md) |
| return false; |
| |
| table = dm_get_live_table(md, &srcu_idx); |
| |
| if (table->num_targets != 1) |
| goto out; |
| |
| ti = dm_table_get_target(table, 0); |
| |
| if (is_trusted_verity_target(ti)) |
| trusted = true; |
| |
| out: |
| dm_put_live_table(md, srcu_idx); |
| dm_put(md); |
| |
| return trusted; |
| } |