2021-08-04 11:41:40 +02:00
// SPDX-License-Identifier: GPL-2.0-only
# include <linux/genhd.h>
2021-09-20 14:33:25 +02:00
# include <linux/slab.h>
2021-08-04 11:41:40 +02:00
struct bd_holder_disk {
struct list_head list ;
2021-08-04 11:41:42 +02:00
struct block_device * bdev ;
2021-08-04 11:41:40 +02:00
int refcnt ;
} ;
static struct bd_holder_disk * bd_find_holder_disk ( struct block_device * bdev ,
struct gendisk * disk )
{
struct bd_holder_disk * holder ;
2021-08-04 11:41:42 +02:00
list_for_each_entry ( holder , & disk - > slave_bdevs , list )
if ( holder - > bdev = = bdev )
2021-08-04 11:41:40 +02:00
return holder ;
return NULL ;
}
static int add_symlink ( struct kobject * from , struct kobject * to )
{
return sysfs_create_link ( from , to , kobject_name ( to ) ) ;
}
static void del_symlink ( struct kobject * from , struct kobject * to )
{
sysfs_remove_link ( from , kobject_name ( to ) ) ;
}
2021-08-04 11:41:43 +02:00
static int __link_disk_holder ( struct block_device * bdev , struct gendisk * disk )
{
int ret ;
ret = add_symlink ( disk - > slave_dir , bdev_kobj ( bdev ) ) ;
if ( ret )
return ret ;
ret = add_symlink ( bdev - > bd_holder_dir , & disk_to_dev ( disk ) - > kobj ) ;
if ( ret )
del_symlink ( disk - > slave_dir , bdev_kobj ( bdev ) ) ;
return ret ;
}
2021-08-04 11:41:40 +02:00
/**
* bd_link_disk_holder - create symlinks between holding disk and slave bdev
* @ bdev : the claimed slave bdev
* @ disk : the holding disk
*
* DON ' T USE THIS UNLESS YOU ' RE ALREADY USING IT .
*
* This functions creates the following sysfs symlinks .
*
* - from " slaves " directory of the holder @ disk to the claimed @ bdev
* - from " holders " directory of the @ bdev to the holder @ disk
*
* For example , if / dev / dm - 0 maps to / dev / sda and disk for dm - 0 is
* passed to bd_link_disk_holder ( ) , then :
*
* / sys / block / dm - 0 / slaves / sda - - > / sys / block / sda
* / sys / block / sda / holders / dm - 0 - - > / sys / block / dm - 0
*
* The caller must have claimed @ bdev before calling this function and
* ensure that both @ bdev and @ disk are valid during the creation and
* lifetime of these symlinks .
*
* CONTEXT :
* Might sleep .
*
* RETURNS :
* 0 on success , - errno on failure .
*/
int bd_link_disk_holder ( struct block_device * bdev , struct gendisk * disk )
{
struct bd_holder_disk * holder ;
int ret = 0 ;
2021-08-04 11:41:42 +02:00
mutex_lock ( & disk - > open_mutex ) ;
2021-08-04 11:41:40 +02:00
WARN_ON_ONCE ( ! bdev - > bd_holder ) ;
/* FIXME: remove the following once add_disk() handles errors */
2021-08-04 11:41:43 +02:00
if ( WARN_ON ( ! bdev - > bd_holder_dir ) )
2021-08-04 11:41:40 +02:00
goto out_unlock ;
holder = bd_find_holder_disk ( bdev , disk ) ;
if ( holder ) {
holder - > refcnt + + ;
goto out_unlock ;
}
holder = kzalloc ( sizeof ( * holder ) , GFP_KERNEL ) ;
if ( ! holder ) {
ret = - ENOMEM ;
goto out_unlock ;
}
INIT_LIST_HEAD ( & holder - > list ) ;
2021-08-04 11:41:42 +02:00
holder - > bdev = bdev ;
2021-08-04 11:41:40 +02:00
holder - > refcnt = 1 ;
2021-08-04 11:41:43 +02:00
if ( disk - > slave_dir ) {
ret = __link_disk_holder ( bdev , disk ) ;
if ( ret ) {
kfree ( holder ) ;
goto out_unlock ;
}
}
2021-08-04 11:41:40 +02:00
2021-08-04 11:41:42 +02:00
list_add ( & holder - > list , & disk - > slave_bdevs ) ;
2021-08-20 11:49:29 +02:00
/*
* del_gendisk drops the initial reference to bd_holder_dir , so we need
* to keep our own here to allow for cleanup past that point .
*/
kobject_get ( bdev - > bd_holder_dir ) ;
2021-08-04 11:41:40 +02:00
out_unlock :
2021-08-04 11:41:42 +02:00
mutex_unlock ( & disk - > open_mutex ) ;
2021-08-04 11:41:40 +02:00
return ret ;
}
EXPORT_SYMBOL_GPL ( bd_link_disk_holder ) ;
2021-08-04 11:41:43 +02:00
static void __unlink_disk_holder ( struct block_device * bdev ,
struct gendisk * disk )
{
del_symlink ( disk - > slave_dir , bdev_kobj ( bdev ) ) ;
del_symlink ( bdev - > bd_holder_dir , & disk_to_dev ( disk ) - > kobj ) ;
}
2021-08-04 11:41:40 +02:00
/**
* bd_unlink_disk_holder - destroy symlinks created by bd_link_disk_holder ( )
* @ bdev : the calimed slave bdev
* @ disk : the holding disk
*
* DON ' T USE THIS UNLESS YOU ' RE ALREADY USING IT .
*
* CONTEXT :
* Might sleep .
*/
void bd_unlink_disk_holder ( struct block_device * bdev , struct gendisk * disk )
{
struct bd_holder_disk * holder ;
2021-08-04 11:41:42 +02:00
mutex_lock ( & disk - > open_mutex ) ;
2021-08-04 11:41:40 +02:00
holder = bd_find_holder_disk ( bdev , disk ) ;
if ( ! WARN_ON_ONCE ( holder = = NULL ) & & ! - - holder - > refcnt ) {
2021-08-04 11:41:43 +02:00
if ( disk - > slave_dir )
__unlink_disk_holder ( bdev , disk ) ;
2021-08-20 11:49:29 +02:00
kobject_put ( bdev - > bd_holder_dir ) ;
2021-08-04 11:41:40 +02:00
list_del_init ( & holder - > list ) ;
kfree ( holder ) ;
}
2021-08-04 11:41:42 +02:00
mutex_unlock ( & disk - > open_mutex ) ;
2021-08-04 11:41:40 +02:00
}
EXPORT_SYMBOL_GPL ( bd_unlink_disk_holder ) ;
2021-08-04 11:41:43 +02:00
int bd_register_pending_holders ( struct gendisk * disk )
{
struct bd_holder_disk * holder ;
int ret ;
mutex_lock ( & disk - > open_mutex ) ;
list_for_each_entry ( holder , & disk - > slave_bdevs , list ) {
ret = __link_disk_holder ( holder - > bdev , disk ) ;
if ( ret )
goto out_undo ;
}
mutex_unlock ( & disk - > open_mutex ) ;
return 0 ;
out_undo :
list_for_each_entry_continue_reverse ( holder , & disk - > slave_bdevs , list )
__unlink_disk_holder ( holder - > bdev , disk ) ;
mutex_unlock ( & disk - > open_mutex ) ;
return ret ;
}