2004-11-18 23:02:21 +03:00
/*
* Copyright ( C ) 2004 Luca Berra
2008-09-19 09:19:09 +04:00
* Copyright ( C ) 2004 - 2008 Red Hat , Inc . All rights reserved .
2004-11-18 23:02:21 +03:00
*
* This file is part of LVM2 .
*
* This copyrighted material is made available to anyone wishing to use ,
* modify , copy , or redistribute it subject to the terms and conditions
* of the GNU Lesser General Public License v .2 .1 .
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program ; if not , write to the Free Software Foundation ,
2016-01-21 13:49:46 +03:00
* Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
2004-11-18 23:02:21 +03:00
*/
2018-05-14 12:30:20 +03:00
# include "lib/misc/lib.h"
# include "lib/device/dev-type.h"
# include "lib/mm/xlate.h"
2019-09-19 19:33:59 +03:00
# include "lib/misc/crc.h"
2014-09-03 17:47:52 +04:00
# ifdef UDEV_SYNC_SUPPORT
# include <libudev.h> /* for MD detection using udev db records */
2018-05-14 12:30:20 +03:00
# include "lib/device/dev-ext-udev-constants.h"
2014-09-03 17:47:52 +04:00
# endif
2004-11-18 23:02:21 +03:00
2013-11-13 17:56:29 +04:00
# ifdef __linux__
2007-10-24 04:30:30 +04:00
2004-11-18 23:02:21 +03:00
/* Lifted from <linux/raid/md_p.h> because of difficulty including it */
# define MD_SB_MAGIC 0xa92b4efc
2007-10-24 15:24:24 +04:00
# define MD_RESERVED_BYTES (64 * 1024ULL)
2004-11-18 23:02:21 +03:00
# define MD_RESERVED_SECTORS (MD_RESERVED_BYTES / 512)
2017-07-19 17:17:30 +03:00
# define MD_NEW_SIZE_SECTORS(x) (((x) & ~(MD_RESERVED_SECTORS - 1)) \
2004-11-18 23:02:21 +03:00
- MD_RESERVED_SECTORS )
2011-07-08 19:53:59 +04:00
# define MD_MAX_SYSFS_SIZE 64
2004-11-18 23:02:21 +03:00
2007-10-24 04:30:30 +04:00
static int _dev_has_md_magic ( struct device * dev , uint64_t sb_offset )
{
uint32_t md_magic ;
/* Version 1 is little endian; version 0.90.0 is machine endian */
2018-05-04 01:12:07 +03:00
if ( ! dev_read_bytes ( dev , sb_offset , sizeof ( uint32_t ) , & md_magic ) )
return_0 ;
if ( ( md_magic = = MD_SB_MAGIC ) | |
( ( MD_SB_MAGIC ! = xlate32 ( MD_SB_MAGIC ) ) & & ( md_magic = = xlate32 ( MD_SB_MAGIC ) ) ) )
2007-10-24 04:30:30 +04:00
return 1 ;
return 0 ;
}
2019-09-19 19:33:59 +03:00
# define IMSM_SIGNATURE "Intel Raid ISM Cfg Sig. "
2020-08-28 19:58:55 +03:00
# define IMSM_SIG_LEN (sizeof(IMSM_SIGNATURE) - 1)
2019-09-19 19:33:59 +03:00
static int _dev_has_imsm_magic ( struct device * dev , uint64_t devsize_sectors )
{
char imsm_signature [ IMSM_SIG_LEN ] ;
uint64_t off = ( devsize_sectors * 512 ) - 1024 ;
if ( ! dev_read_bytes ( dev , off , IMSM_SIG_LEN , imsm_signature ) )
return_0 ;
if ( ! memcmp ( imsm_signature , IMSM_SIGNATURE , IMSM_SIG_LEN ) )
return 1 ;
return 0 ;
}
# define DDF_MAGIC 0xDE11DE11
struct ddf_header {
uint32_t magic ;
uint32_t crc ;
char guid [ 24 ] ;
char revision [ 8 ] ;
char padding [ 472 ] ;
} ;
2019-09-19 20:17:36 +03:00
static int _dev_has_ddf_magic ( struct device * dev , uint64_t devsize_sectors , uint64_t * sb_offset )
2019-09-19 19:33:59 +03:00
{
struct ddf_header hdr ;
uint32_t crc , our_crc ;
uint64_t off ;
uint64_t devsize_bytes = devsize_sectors * 512 ;
if ( devsize_bytes < 0x30000 )
return 0 ;
/* 512 bytes before the end of device (from libblkid) */
off = ( ( devsize_bytes / 0x200 ) - 1 ) * 0x200 ;
if ( ! dev_read_bytes ( dev , off , 512 , & hdr ) )
return_0 ;
if ( ( hdr . magic = = cpu_to_be32 ( DDF_MAGIC ) ) | |
( hdr . magic = = cpu_to_le32 ( DDF_MAGIC ) ) ) {
crc = hdr . crc ;
hdr . crc = 0xffffffff ;
our_crc = calc_crc ( 0 , ( const uint8_t * ) & hdr , 512 ) ;
if ( ( cpu_to_be32 ( our_crc ) = = crc ) | |
( cpu_to_le32 ( our_crc ) = = crc ) ) {
2019-09-19 20:17:36 +03:00
* sb_offset = off ;
2019-09-19 19:33:59 +03:00
return 1 ;
} else {
log_debug_devs ( " Found md ddf magic at %llu wrong crc %x disk %x %s " ,
( unsigned long long ) off , our_crc , crc , dev_name ( dev ) ) ;
return 0 ;
}
}
/* 128KB before the end of device (from libblkid) */
off = ( ( devsize_bytes / 0x200 ) - 257 ) * 0x200 ;
if ( ! dev_read_bytes ( dev , off , 512 , & hdr ) )
return_0 ;
if ( ( hdr . magic = = cpu_to_be32 ( DDF_MAGIC ) ) | |
( hdr . magic = = cpu_to_le32 ( DDF_MAGIC ) ) ) {
crc = hdr . crc ;
hdr . crc = 0xffffffff ;
our_crc = calc_crc ( 0 , ( const uint8_t * ) & hdr , 512 ) ;
if ( ( cpu_to_be32 ( our_crc ) = = crc ) | |
( cpu_to_le32 ( our_crc ) = = crc ) ) {
2019-09-19 20:17:36 +03:00
* sb_offset = off ;
2019-09-19 19:33:59 +03:00
return 1 ;
} else {
log_debug_devs ( " Found md ddf magic at %llu wrong crc %x disk %x %s " ,
( unsigned long long ) off , our_crc , crc , dev_name ( dev ) ) ;
return 0 ;
}
}
return 0 ;
}
2019-05-21 19:44:39 +03:00
/*
* _udev_dev_is_md_component ( ) only works if
* external_device_info_source = " udev "
*
* but
*
* udev_dev_is_md_component ( ) in dev - type . c only works if
* obtain_device_list_from_udev = 1
*
* and neither of those config setting matches very well
* with what we ' re doing here .
*/
2014-09-03 17:47:52 +04:00
# ifdef UDEV_SYNC_SUPPORT
2019-05-21 19:44:39 +03:00
static int _udev_dev_is_md_component ( struct device * dev )
2014-09-03 17:47:52 +04:00
{
const char * value ;
struct dev_ext * ext ;
if ( ! ( ext = dev_ext_get ( dev ) ) )
return_0 ;
2019-05-21 20:06:34 +03:00
if ( ! ( value = udev_device_get_property_value ( ( struct udev_device * ) ext - > handle , DEV_EXT_UDEV_BLKID_TYPE ) ) ) {
dev - > flags | = DEV_UDEV_INFO_MISSING ;
2014-09-03 17:47:52 +04:00
return 0 ;
2019-05-21 20:06:34 +03:00
}
2014-09-03 17:47:52 +04:00
2015-01-29 18:44:34 +03:00
return ! strcmp ( value , DEV_EXT_UDEV_BLKID_TYPE_SW_RAID ) ;
2014-09-03 17:47:52 +04:00
}
# else
2019-05-21 19:44:39 +03:00
static int _udev_dev_is_md_component ( struct device * dev )
2014-09-03 17:47:52 +04:00
{
2019-05-21 20:06:34 +03:00
dev - > flags | = DEV_UDEV_INFO_MISSING ;
2014-09-03 17:47:52 +04:00
return 0 ;
}
# endif
2004-11-18 23:02:21 +03:00
/*
* Returns - 1 on error
*/
2019-05-21 19:44:39 +03:00
static int _native_dev_is_md_component ( struct device * dev , uint64_t * offset_found , int full )
2004-11-18 23:02:21 +03:00
{
2020-08-29 21:27:32 +03:00
uint64_t size , sb_offset = 0 ;
2018-05-04 01:12:07 +03:00
int ret ;
if ( ! scan_bcache )
return - EAGAIN ;
2004-11-18 23:02:21 +03:00
if ( ! dev_get_size ( dev , & size ) ) {
stack ;
return - 1 ;
}
if ( size < MD_RESERVED_SECTORS * 2 )
return 0 ;
2018-05-04 01:12:07 +03:00
/*
2019-09-19 20:17:36 +03:00
* Some md versions locate the magic number at the end of the device .
* Those checks can ' t be satisfied with the initial scan data , and
* require an extra read i / o at the end of every device . Issuing
2018-05-04 01:12:07 +03:00
* an extra read to every device in every command , just to check for
2018-05-11 23:52:22 +03:00
* the old md format is a bad tradeoff .
2018-05-04 01:12:07 +03:00
*
* When " full " is set , we check a the start and end of the device for
* md magic numbers . When " full " is not set , we only check at the
* start of the device for the magic numbers . We decide for each
* command if it should do a full check ( cmd - > use_full_md_check ) ,
* and set it for commands that could possibly write to an md dev
* ( pvcreate / vgcreate / vgextend ) .
*/
2019-09-19 20:17:36 +03:00
/*
* md superblock version 1.1 at offset 0 from start
*/
if ( _dev_has_md_magic ( dev , 0 ) ) {
log_debug_devs ( " Found md magic number at offset 0 of %s. " , dev_name ( dev ) ) ;
ret = 1 ;
goto out ;
}
2018-05-04 01:12:07 +03:00
2019-09-19 20:17:36 +03:00
/*
* md superblock version 1.2 at offset 4 KB from start
*/
if ( _dev_has_md_magic ( dev , 4096 ) ) {
log_debug_devs ( " Found md magic number at offset 4096 of %s. " , dev_name ( dev ) ) ;
ret = 1 ;
goto out ;
}
if ( ! full ) {
2018-05-04 01:12:07 +03:00
ret = 0 ;
goto out ;
2004-11-18 23:02:21 +03:00
}
2019-09-19 20:17:36 +03:00
/*
* Handle superblocks at the end of the device .
*/
/*
* md superblock version 0 at 64 KB from end of device
* ( after end is aligned to 64 KB )
*/
2004-11-18 23:02:21 +03:00
sb_offset = MD_NEW_SIZE_SECTORS ( size ) < < SECTOR_SHIFT ;
2019-09-19 20:17:36 +03:00
2018-05-04 01:12:07 +03:00
if ( _dev_has_md_magic ( dev , sb_offset ) ) {
2019-09-19 20:17:36 +03:00
log_debug_devs ( " Found md magic number at offset %llu of %s. " , ( unsigned long long ) sb_offset , dev_name ( dev ) ) ;
2018-05-04 01:12:07 +03:00
ret = 1 ;
2007-10-24 04:51:05 +04:00
goto out ;
2018-05-04 01:12:07 +03:00
}
2004-11-18 23:02:21 +03:00
2019-09-19 20:17:36 +03:00
/*
* md superblock version 1.0 at 8 KB from end of device
*/
sb_offset = ( ( size - 8 * 2 ) & ~ ( 4 * 2 - 1ULL ) ) < < SECTOR_SHIFT ;
2019-09-19 19:33:59 +03:00
2019-09-19 20:17:36 +03:00
if ( _dev_has_md_magic ( dev , sb_offset ) ) {
log_debug_devs ( " Found md magic number at offset %llu of %s. " , ( unsigned long long ) sb_offset , dev_name ( dev ) ) ;
ret = 1 ;
goto out ;
}
2019-09-19 19:33:59 +03:00
2019-09-19 20:17:36 +03:00
/*
* md imsm superblock 1 K from end of device
*/
2007-10-24 04:51:05 +04:00
2019-09-19 19:33:59 +03:00
if ( _dev_has_imsm_magic ( dev , size ) ) {
2019-09-19 20:17:36 +03:00
log_debug_devs ( " Found md imsm magic number at offset %llu of %s. " , ( unsigned long long ) sb_offset , dev_name ( dev ) ) ;
sb_offset = 1024 ;
2019-09-19 19:33:59 +03:00
ret = 1 ;
goto out ;
}
2019-09-19 20:17:36 +03:00
/*
* md ddf superblock 512 bytes from end , or 128 KB from end
*/
if ( _dev_has_ddf_magic ( dev , size , & sb_offset ) ) {
log_debug_devs ( " Found md ddf magic number at offset %llu of %s. " , ( unsigned long long ) sb_offset , dev_name ( dev ) ) ;
2019-09-19 19:33:59 +03:00
ret = 1 ;
goto out ;
}
2007-10-24 04:51:05 +04:00
ret = 0 ;
out :
2013-11-06 18:09:29 +04:00
if ( ret & & offset_found )
* offset_found = sb_offset ;
2007-10-24 04:51:05 +04:00
2004-11-18 23:02:21 +03:00
return ret ;
}
2019-05-21 19:44:39 +03:00
int dev_is_md_component ( struct device * dev , uint64_t * offset_found , int full )
2014-09-03 17:47:52 +04:00
{
2018-12-03 20:22:45 +03:00
int ret ;
2014-09-03 17:47:52 +04:00
/*
* If non - native device status source is selected , use it
* only if offset_found is not requested as this
* information is not in udev db .
*/
2018-12-03 20:22:45 +03:00
if ( ( dev - > ext . src = = DEV_EXT_NONE ) | | offset_found ) {
2019-05-21 19:44:39 +03:00
ret = _native_dev_is_md_component ( dev , offset_found , full ) ;
2018-12-03 20:22:45 +03:00
if ( ! full ) {
if ( ! ret | | ( ret = = - EAGAIN ) ) {
if ( udev_dev_is_md_component ( dev ) )
2019-05-21 19:58:01 +03:00
ret = 1 ;
2018-12-03 20:22:45 +03:00
}
}
2019-05-21 19:58:01 +03:00
if ( ret & & ( ret ! = - EAGAIN ) )
dev - > flags | = DEV_IS_MD_COMPONENT ;
2018-12-03 20:22:45 +03:00
return ret ;
}
2014-09-03 17:47:52 +04:00
2019-05-21 19:58:01 +03:00
if ( dev - > ext . src = = DEV_EXT_UDEV ) {
ret = _udev_dev_is_md_component ( dev ) ;
if ( ret & & ( ret ! = - EAGAIN ) )
dev - > flags | = DEV_IS_MD_COMPONENT ;
return ret ;
}
2014-09-03 17:47:52 +04:00
log_error ( INTERNAL_ERROR " Missing hook for MD device recognition "
" using external device info source %s " , dev_ext_name ( dev ) ) ;
return - 1 ;
}
2009-07-06 23:04:24 +04:00
static int _md_sysfs_attribute_snprintf ( char * path , size_t size ,
2013-06-12 14:08:56 +04:00
struct dev_types * dt ,
2009-08-01 21:11:02 +04:00
struct device * blkdev ,
2009-07-06 23:04:24 +04:00
const char * attribute )
2008-09-19 09:19:09 +04:00
{
2013-06-12 13:38:48 +04:00
const char * sysfs_dir = dm_sysfs_dir ( ) ;
2008-09-19 09:19:09 +04:00
struct stat info ;
2009-08-01 21:11:02 +04:00
dev_t dev = blkdev - > dev ;
2009-07-06 23:04:24 +04:00
int ret = - 1 ;
2008-09-19 09:19:09 +04:00
2009-08-01 21:11:02 +04:00
if ( ! sysfs_dir | | ! * sysfs_dir )
2009-07-06 23:04:24 +04:00
return ret ;
2008-09-19 09:19:09 +04:00
2013-06-12 14:08:56 +04:00
if ( MAJOR ( dev ) = = dt - > blkext_major ) {
2009-08-19 19:34:33 +04:00
/* lookup parent MD device from blkext partition */
2013-06-12 16:33:28 +04:00
if ( ! dev_get_primary_dev ( dt , blkdev , & dev ) )
2009-08-19 19:34:33 +04:00
return ret ;
2009-08-01 21:11:02 +04:00
}
2008-09-19 09:19:09 +04:00
2013-06-12 14:08:56 +04:00
if ( MAJOR ( dev ) ! = dt - > md_major )
2009-08-19 19:34:33 +04:00
return ret ;
2009-07-20 22:33:16 +04:00
ret = dm_snprintf ( path , size , " %s/dev/block/%d:%d/md/%s " , sysfs_dir ,
2009-08-01 21:11:02 +04:00
( int ) MAJOR ( dev ) , ( int ) MINOR ( dev ) , attribute ) ;
2009-07-06 23:04:24 +04:00
if ( ret < 0 ) {
log_error ( " dm_snprintf md %s failed " , attribute ) ;
return ret ;
2008-09-19 09:19:09 +04:00
}
2009-08-01 21:14:52 +04:00
if ( stat ( path , & info ) = = - 1 ) {
if ( errno ! = ENOENT ) {
log_sys_error ( " stat " , path ) ;
return ret ;
}
2009-07-06 23:04:24 +04:00
/* old sysfs structure */
ret = dm_snprintf ( path , size , " %s/block/md%d/md/%s " ,
2009-08-01 21:11:02 +04:00
sysfs_dir , ( int ) MINOR ( dev ) , attribute ) ;
2009-07-06 23:04:24 +04:00
if ( ret < 0 ) {
log_error ( " dm_snprintf old md %s failed " , attribute ) ;
return ret ;
}
2008-09-19 09:19:09 +04:00
}
2009-07-06 23:04:24 +04:00
return ret ;
}
2013-06-12 14:08:56 +04:00
static int _md_sysfs_attribute_scanf ( struct dev_types * dt ,
2009-07-06 23:04:24 +04:00
struct device * dev ,
const char * attribute_name ,
const char * attribute_fmt ,
void * attribute_value )
{
2011-07-08 19:53:59 +04:00
char path [ PATH_MAX + 1 ] , buffer [ MD_MAX_SYSFS_SIZE ] ;
2009-07-06 23:04:24 +04:00
FILE * fp ;
int ret = 0 ;
2013-06-12 14:08:56 +04:00
if ( _md_sysfs_attribute_snprintf ( path , PATH_MAX , dt ,
dev , attribute_name ) < 0 )
2009-07-06 23:04:24 +04:00
return ret ;
2008-09-19 09:19:09 +04:00
if ( ! ( fp = fopen ( path , " r " ) ) ) {
2019-07-09 22:48:31 +03:00
log_debug ( " _md_sysfs_attribute_scanf fopen failed %s " , path ) ;
2009-07-06 23:04:24 +04:00
return ret ;
2008-09-19 09:19:09 +04:00
}
if ( ! fgets ( buffer , sizeof ( buffer ) , fp ) ) {
2019-07-09 22:48:31 +03:00
log_debug ( " _md_sysfs_attribute_scanf fgets failed %s " , path ) ;
2008-09-19 09:19:09 +04:00
goto out ;
}
2009-07-06 23:04:24 +04:00
if ( ( ret = sscanf ( buffer , attribute_fmt , attribute_value ) ) ! = 1 ) {
log_error ( " %s sysfs attr %s not in expected format: %s " ,
dev_name ( dev ) , attribute_name , buffer ) ;
2008-09-19 09:19:09 +04:00
goto out ;
}
out :
if ( fclose ( fp ) )
log_sys_error ( " fclose " , path ) ;
2009-07-06 23:04:24 +04:00
return ret ;
}
/*
* Retrieve chunk size from md device using sysfs .
*/
2017-10-18 17:57:46 +03:00
static unsigned long _dev_md_chunk_size ( struct dev_types * dt , struct device * dev )
2009-07-06 23:04:24 +04:00
{
const char * attribute = " chunk_size " ;
unsigned long chunk_size_bytes = 0UL ;
2013-06-12 14:08:56 +04:00
if ( _md_sysfs_attribute_scanf ( dt , dev , attribute ,
2009-07-06 23:04:24 +04:00
" %lu " , & chunk_size_bytes ) ! = 1 )
return 0 ;
log_very_verbose ( " Device %s %s is %lu bytes. " ,
dev_name ( dev ) , attribute , chunk_size_bytes ) ;
2008-10-03 18:22:18 +04:00
return chunk_size_bytes > > SECTOR_SHIFT ;
2008-09-19 09:19:09 +04:00
}
2009-07-06 23:04:24 +04:00
/*
* Retrieve level from md device using sysfs .
*/
2017-10-18 17:57:46 +03:00
static int _dev_md_level ( struct dev_types * dt , struct device * dev )
2009-07-06 23:04:24 +04:00
{
2011-07-08 19:53:59 +04:00
char level_string [ MD_MAX_SYSFS_SIZE ] ;
2009-07-06 23:04:24 +04:00
const char * attribute = " level " ;
int level = - 1 ;
2013-06-12 14:08:56 +04:00
if ( _md_sysfs_attribute_scanf ( dt , dev , attribute ,
2011-07-08 19:53:59 +04:00
" %s " , & level_string ) ! = 1 )
2009-07-06 23:04:24 +04:00
return - 1 ;
2011-07-08 19:53:59 +04:00
log_very_verbose ( " Device %s %s is %s. " ,
dev_name ( dev ) , attribute , level_string ) ;
/* We only care about raid - ignore linear/faulty/multipath etc. */
if ( sscanf ( level_string , " raid%d " , & level ) ! = 1 )
return - 1 ;
2009-07-06 23:04:24 +04:00
return level ;
}
/*
* Retrieve raid_disks from md device using sysfs .
*/
2017-10-18 17:57:46 +03:00
static int _dev_md_raid_disks ( struct dev_types * dt , struct device * dev )
2009-07-06 23:04:24 +04:00
{
const char * attribute = " raid_disks " ;
int raid_disks = 0 ;
2013-06-12 14:08:56 +04:00
if ( _md_sysfs_attribute_scanf ( dt , dev , attribute ,
2009-07-06 23:04:24 +04:00
" %d " , & raid_disks ) ! = 1 )
return 0 ;
log_very_verbose ( " Device %s %s is %d. " ,
dev_name ( dev ) , attribute , raid_disks ) ;
return raid_disks ;
}
/*
* Calculate stripe width of md device using its sysfs files .
*/
2013-06-12 14:08:56 +04:00
unsigned long dev_md_stripe_width ( struct dev_types * dt , struct device * dev )
2009-07-06 23:04:24 +04:00
{
unsigned long chunk_size_sectors = 0UL ;
unsigned long stripe_width_sectors = 0UL ;
int level , raid_disks , data_disks ;
2017-10-18 17:57:46 +03:00
chunk_size_sectors = _dev_md_chunk_size ( dt , dev ) ;
2009-07-06 23:04:24 +04:00
if ( ! chunk_size_sectors )
return 0 ;
2017-10-18 17:57:46 +03:00
level = _dev_md_level ( dt , dev ) ;
2009-07-06 23:04:24 +04:00
if ( level < 0 )
return 0 ;
2017-10-18 17:57:46 +03:00
raid_disks = _dev_md_raid_disks ( dt , dev ) ;
2009-07-06 23:04:24 +04:00
if ( ! raid_disks )
return 0 ;
/* The raid level governs the number of data disks. */
switch ( level ) {
case 0 :
/* striped md does not have any parity disks */
data_disks = raid_disks ;
break ;
case 1 :
case 10 :
/* mirrored md effectively has 1 data disk */
data_disks = 1 ;
break ;
case 4 :
case 5 :
/* both raid 4 and 5 have a single parity disk */
data_disks = raid_disks - 1 ;
break ;
case 6 :
/* raid 6 has 2 parity disks */
data_disks = raid_disks - 2 ;
break ;
default :
log_error ( " Device %s has an unknown md raid level: %d " ,
dev_name ( dev ) , level ) ;
return 0 ;
}
stripe_width_sectors = chunk_size_sectors * data_disks ;
log_very_verbose ( " Device %s stripe-width is %lu bytes. " ,
dev_name ( dev ) ,
stripe_width_sectors < < SECTOR_SHIFT ) ;
return stripe_width_sectors ;
}
2018-06-15 19:42:10 +03:00
int dev_is_md_with_end_superblock ( struct dev_types * dt , struct device * dev )
{
char version_string [ MD_MAX_SYSFS_SIZE ] ;
const char * attribute = " metadata_version " ;
if ( MAJOR ( dev - > dev ) ! = dt - > md_major )
return 0 ;
if ( _md_sysfs_attribute_scanf ( dt , dev , attribute ,
" %s " , & version_string ) ! = 1 )
2019-07-09 22:48:31 +03:00
return 0 ;
2018-06-15 19:42:10 +03:00
log_very_verbose ( " Device %s %s is %s. " ,
dev_name ( dev ) , attribute , version_string ) ;
2018-11-29 21:35:54 +03:00
if ( ! strcmp ( version_string , " 1.0 " ) | | ! strcmp ( version_string , " 0.90 " ) )
2018-06-15 19:42:10 +03:00
return 1 ;
return 0 ;
}
2007-10-24 04:30:30 +04:00
# else
2019-05-21 19:44:39 +03:00
int dev_is_md_component ( struct device * dev __attribute__ ( ( unused ) ) ,
2010-07-09 19:34:40 +04:00
uint64_t * sb __attribute__ ( ( unused ) ) )
2007-10-24 04:30:30 +04:00
{
return 0 ;
}
2013-06-12 14:08:56 +04:00
unsigned long dev_md_stripe_width ( struct dev_types * dt __attribute__ ( ( unused ) ) ,
struct device * dev __attribute__ ( ( unused ) ) )
2008-09-19 09:19:09 +04:00
{
return 0UL ;
}
2007-10-24 04:30:30 +04:00
# endif