2004-11-18 20:02:21 +00:00
/*
* Copyright ( C ) 2004 Luca Berra
2008-09-19 05:19:09 +00:00
* Copyright ( C ) 2004 - 2008 Red Hat , Inc . All rights reserved .
2004-11-18 20:02:21 +00: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 ,
* Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include "lib.h"
# include "metadata.h"
2004-11-27 22:07:41 +00:00
# include "xlate.h"
2008-09-19 05:19:09 +00:00
# include "filter.h"
2004-11-18 20:02:21 +00:00
2007-10-24 00:30:30 +00:00
# ifdef linux
2004-11-18 20:02:21 +00:00
/* Lifted from <linux/raid/md_p.h> because of difficulty including it */
# define MD_SB_MAGIC 0xa92b4efc
2007-10-24 11:24:24 +00:00
# define MD_RESERVED_BYTES (64 * 1024ULL)
2004-11-18 20:02:21 +00:00
# define MD_RESERVED_SECTORS (MD_RESERVED_BYTES / 512)
# define MD_NEW_SIZE_SECTORS(x) ((x & ~(MD_RESERVED_SECTORS - 1)) \
- MD_RESERVED_SECTORS )
2007-10-24 00:30:30 +00: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 */
if ( dev_read ( dev , sb_offset , sizeof ( uint32_t ) , & md_magic ) & &
( ( md_magic = = xlate32 ( MD_SB_MAGIC ) ) | |
( md_magic = = MD_SB_MAGIC ) ) )
return 1 ;
return 0 ;
}
2007-10-24 11:24:24 +00:00
/*
* Calculate the position of the superblock .
* It is always aligned to a 4 K boundary and
* depending on minor_version , it can be :
* 0 : At least 8 K , but less than 12 K , from end of device
* 1 : At start of device
* 2 : 4 K from start of device .
*/
2008-06-23 14:54:50 +00:00
typedef enum {
MD_MINOR_VERSION_MIN ,
MD_MINOR_V0 = MD_MINOR_VERSION_MIN ,
MD_MINOR_V1 ,
MD_MINOR_V2 ,
MD_MINOR_VERSION_MAX = MD_MINOR_V2
} md_minor_version_t ;
static uint64_t _v1_sb_offset ( uint64_t size , md_minor_version_t minor_version )
2007-10-24 00:51:05 +00:00
{
2008-06-23 19:26:21 +00:00
uint64_t uninitialized_var ( sb_offset ) ;
2007-10-24 00:51:05 +00:00
switch ( minor_version ) {
2008-06-23 14:54:50 +00:00
case MD_MINOR_V0 :
2007-10-24 11:24:24 +00:00
sb_offset = ( size - 8 * 2 ) & ~ ( 4 * 2 - 1ULL ) ;
2007-10-24 00:51:05 +00:00
break ;
2008-06-23 14:54:50 +00:00
case MD_MINOR_V1 :
2007-10-24 00:51:05 +00:00
sb_offset = 0 ;
break ;
2008-06-23 14:54:50 +00:00
case MD_MINOR_V2 :
2007-10-24 00:51:05 +00:00
sb_offset = 4 * 2 ;
break ;
}
sb_offset < < = SECTOR_SHIFT ;
return sb_offset ;
}
2004-11-18 20:02:21 +00:00
/*
* Returns - 1 on error
*/
int dev_is_md ( struct device * dev , uint64_t * sb )
{
2007-10-24 00:51:05 +00:00
int ret = 1 ;
2008-06-23 14:54:50 +00:00
md_minor_version_t minor ;
2004-11-18 20:02:21 +00:00
uint64_t size , sb_offset ;
if ( ! dev_get_size ( dev , & size ) ) {
stack ;
return - 1 ;
}
if ( size < MD_RESERVED_SECTORS * 2 )
return 0 ;
if ( ! dev_open ( dev ) ) {
stack ;
return - 1 ;
}
2007-10-24 00:51:05 +00:00
/* Check if it is an md component device. */
/* Version 0.90.0 */
2004-11-18 20:02:21 +00:00
sb_offset = MD_NEW_SIZE_SECTORS ( size ) < < SECTOR_SHIFT ;
2007-10-24 00:51:05 +00:00
if ( _dev_has_md_magic ( dev , sb_offset ) )
goto out ;
2004-11-18 20:02:21 +00:00
2008-06-23 14:54:50 +00:00
minor = MD_MINOR_VERSION_MIN ;
2007-10-24 00:51:05 +00:00
/* Version 1, try v1.0 -> v1.2 */
do {
sb_offset = _v1_sb_offset ( size , minor ) ;
if ( _dev_has_md_magic ( dev , sb_offset ) )
goto out ;
2008-06-23 14:54:50 +00:00
} while ( + + minor < = MD_MINOR_VERSION_MAX ) ;
2007-10-24 00:51:05 +00:00
ret = 0 ;
2004-11-18 20:02:21 +00:00
2007-10-24 00:51:05 +00:00
out :
2004-11-18 20:02:21 +00:00
if ( ! dev_close ( dev ) )
stack ;
2007-10-24 00:51:05 +00:00
if ( ret & & sb )
* sb = sb_offset ;
2004-11-18 20:02:21 +00:00
return ret ;
}
2009-07-06 19:04:24 +00:00
static int _md_sysfs_attribute_snprintf ( char * path , size_t size ,
const char * sysfs_dir ,
struct device * dev ,
const char * attribute )
2008-09-19 05:19:09 +00:00
{
struct stat info ;
2009-07-06 19:04:24 +00:00
int ret = - 1 ;
2008-09-19 05:19:09 +00:00
if ( MAJOR ( dev - > dev ) ! = md_major ( ) )
2009-07-06 19:04:24 +00:00
return ret ;
2008-09-19 05:19:09 +00:00
if ( ! sysfs_dir | | ! * sysfs_dir )
2009-07-06 19:04:24 +00:00
return ret ;
2008-09-19 05:19:09 +00:00
2009-07-06 19:04:24 +00:00
ret = dm_snprintf ( path , size , " %s/dev/block/%d:%d/md/%s " ,
sysfs_dir , MAJOR ( dev - > dev ) , MINOR ( dev - > dev ) , attribute ) ;
if ( ret < 0 ) {
log_error ( " dm_snprintf md %s failed " , attribute ) ;
return ret ;
2008-09-19 05:19:09 +00:00
}
2009-07-06 19:04:24 +00:00
if ( stat ( path , & info ) < 0 ) {
/* old sysfs structure */
ret = dm_snprintf ( path , size , " %s/block/md%d/md/%s " ,
sysfs_dir , MINOR ( dev - > dev ) , attribute ) ;
if ( ret < 0 ) {
log_error ( " dm_snprintf old md %s failed " , attribute ) ;
return ret ;
}
2008-09-19 05:19:09 +00:00
}
2009-07-06 19:04:24 +00:00
return ret ;
}
static int _md_sysfs_attribute_scanf ( const char * sysfs_dir ,
struct device * dev ,
const char * attribute_name ,
const char * attribute_fmt ,
void * attribute_value )
{
char path [ PATH_MAX + 1 ] , buffer [ 64 ] ;
FILE * fp ;
int ret = 0 ;
if ( _md_sysfs_attribute_snprintf ( path , PATH_MAX , sysfs_dir ,
dev , attribute_name ) < 0 )
return ret ;
2008-09-19 05:19:09 +00:00
if ( ! ( fp = fopen ( path , " r " ) ) ) {
log_sys_error ( " fopen " , path ) ;
2009-07-06 19:04:24 +00:00
return ret ;
2008-09-19 05:19:09 +00:00
}
if ( ! fgets ( buffer , sizeof ( buffer ) , fp ) ) {
log_sys_error ( " fgets " , path ) ;
goto out ;
}
2009-07-06 19:04:24 +00: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 05:19:09 +00:00
goto out ;
}
out :
if ( fclose ( fp ) )
log_sys_error ( " fclose " , path ) ;
2009-07-06 19:04:24 +00:00
return ret ;
}
/*
* Retrieve chunk size from md device using sysfs .
*/
static unsigned long dev_md_chunk_size ( const char * sysfs_dir ,
struct device * dev )
{
const char * attribute = " chunk_size " ;
unsigned long chunk_size_bytes = 0UL ;
if ( _md_sysfs_attribute_scanf ( sysfs_dir , dev , attribute ,
" %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 14:22:18 +00:00
return chunk_size_bytes > > SECTOR_SHIFT ;
2008-09-19 05:19:09 +00:00
}
2009-07-06 19:04:24 +00:00
/*
* Retrieve level from md device using sysfs .
*/
static int dev_md_level ( const char * sysfs_dir , struct device * dev )
{
const char * attribute = " level " ;
int level = - 1 ;
if ( _md_sysfs_attribute_scanf ( sysfs_dir , dev , attribute ,
" raid%d " , & level ) ! = 1 )
return - 1 ;
log_very_verbose ( " Device %s %s is raid%d. " ,
dev_name ( dev ) , attribute , level ) ;
return level ;
}
/*
* Retrieve raid_disks from md device using sysfs .
*/
static int dev_md_raid_disks ( const char * sysfs_dir , struct device * dev )
{
const char * attribute = " raid_disks " ;
int raid_disks = 0 ;
if ( _md_sysfs_attribute_scanf ( sysfs_dir , dev , attribute ,
" %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 .
*/
unsigned long dev_md_stripe_width ( const char * sysfs_dir , struct device * dev )
{
unsigned long chunk_size_sectors = 0UL ;
unsigned long stripe_width_sectors = 0UL ;
int level , raid_disks , data_disks ;
chunk_size_sectors = dev_md_chunk_size ( sysfs_dir , dev ) ;
if ( ! chunk_size_sectors )
return 0 ;
level = dev_md_level ( sysfs_dir , dev ) ;
if ( level < 0 )
return 0 ;
raid_disks = dev_md_raid_disks ( sysfs_dir , dev ) ;
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 ;
}
2007-10-24 00:30:30 +00:00
# else
2007-10-24 00:51:05 +00:00
int dev_is_md ( struct device * dev __attribute ( ( unused ) ) ,
uint64_t * sb __attribute ( ( unused ) ) )
2007-10-24 00:30:30 +00:00
{
return 0 ;
}
2009-07-06 19:04:24 +00:00
unsigned long dev_md_stripe_width ( const char * sysfs_dir __attribute ( ( unused ) ) ,
struct device * dev __attribute ( ( unused ) ) )
2008-09-19 05:19:09 +00:00
{
return 0UL ;
}
2007-10-24 00:30:30 +00:00
# endif