2022-06-27 08:35:24 -07:00
// SPDX-License-Identifier: GPL-2.0-only
# include <linux/list.h>
# include <linux/kernel.h>
# include <linux/dm-verity-loadpin.h>
# include "dm.h"
2022-07-28 08:54:41 -07:00
# include "dm-core.h"
2022-06-27 08:35:24 -07:00
# 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 )
{
2022-09-07 13:30:58 -07:00
int verity_mode ;
2022-06-27 08:35:24 -07:00
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 ;
2022-09-07 13:30:58 -07:00
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 ;
2022-06-27 08:35:24 -07:00
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 ;
2023-06-27 20:28:01 +00:00
if ( bdev = = NULL )
return false ;
2022-06-27 08:35:24 -07:00
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 ) ;
2022-07-28 08:54:41 -07:00
if ( table - > num_targets ! = 1 )
2022-06-27 08:35:24 -07:00
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 ;
}