2005-04-16 15:20:36 -07:00
/*
* Author ( s ) . . . . . . : Holger Smolinski < Holger . Smolinski @ de . ibm . com >
* Volker Sameske < sameske @ de . ibm . com >
* Bugreports . to . . : < Linux390 @ de . ibm . com >
2012-08-31 10:52:13 +02:00
* Copyright IBM Corp . 1999 , 2012
2005-04-16 15:20:36 -07:00
*/
# include <linux/buffer_head.h>
# include <linux/hdreg.h>
# include <linux/slab.h>
# include <asm/dasd.h>
# include <asm/ebcdic.h>
2016-12-24 11:46:01 -08:00
# include <linux/uaccess.h>
2005-04-16 15:20:36 -07:00
# include <asm/vtoc.h>
# include "check.h"
# include "ibm.h"
2012-08-31 10:52:13 +02:00
union label_t {
struct vtoc_volume_label_cdl vol ;
struct vtoc_volume_label_ldl lnx ;
struct vtoc_cms_label cms ;
} ;
2005-04-16 15:20:36 -07:00
/*
2006-03-24 03:15:23 -08:00
* compute the block number from a
2005-04-16 15:20:36 -07:00
* cyl - cyl - head - head structure
*/
2012-08-31 10:52:13 +02:00
static sector_t cchh2blk ( struct vtoc_cchh * ptr , struct hd_geometry * geo )
{
2009-03-26 15:23:47 +01:00
sector_t cyl ;
__u16 head ;
2012-08-31 10:52:13 +02:00
/* decode cylinder and heads for large volumes */
2009-03-26 15:23:47 +01:00
cyl = ptr - > hh & 0xFFF0 ;
cyl < < = 12 ;
cyl | = ptr - > cc ;
head = ptr - > hh & 0x000F ;
return cyl * geo - > heads * geo - > sectors +
head * geo - > sectors ;
2005-04-16 15:20:36 -07:00
}
/*
2006-03-24 03:15:23 -08:00
* compute the block number from a
2005-04-16 15:20:36 -07:00
* cyl - cyl - head - head - block structure
*/
2012-08-31 10:52:13 +02:00
static sector_t cchhb2blk ( struct vtoc_cchhb * ptr , struct hd_geometry * geo )
{
2009-03-26 15:23:47 +01:00
sector_t cyl ;
__u16 head ;
2012-08-31 10:52:13 +02:00
/* decode cylinder and heads for large volumes */
2009-03-26 15:23:47 +01:00
cyl = ptr - > hh & 0xFFF0 ;
cyl < < = 12 ;
cyl | = ptr - > cc ;
head = ptr - > hh & 0x000F ;
return cyl * geo - > heads * geo - > sectors +
head * geo - > sectors +
2005-04-16 15:20:36 -07:00
ptr - > b ;
}
2012-08-31 10:52:13 +02:00
static int find_label ( struct parsed_partitions * state ,
dasd_information2_t * info ,
struct hd_geometry * geo ,
int blocksize ,
sector_t * labelsect ,
char name [ ] ,
char type [ ] ,
union label_t * label )
{
Sector sect ;
unsigned char * data ;
sector_t testsect [ 3 ] ;
unsigned char temp [ 5 ] ;
int found = 0 ;
int i , testcount ;
/* There a three places where we may find a valid label:
* - on an ECKD disk it ' s block 2
* - on an FBA disk it ' s block 1
* - on an CMS formatted FBA disk it is sector 1 , even if the block size
* is larger than 512 bytes ( possible if the DIAG discipline is used )
* If we have a valid info structure , then we know exactly which case we
* have , otherwise we just search through all possebilities .
*/
if ( info ) {
if ( ( info - > cu_type = = 0x6310 & & info - > dev_type = = 0x9336 ) | |
( info - > cu_type = = 0x3880 & & info - > dev_type = = 0x3370 ) )
testsect [ 0 ] = info - > label_block ;
else
testsect [ 0 ] = info - > label_block * ( blocksize > > 9 ) ;
testcount = 1 ;
} else {
testsect [ 0 ] = 1 ;
testsect [ 1 ] = ( blocksize > > 9 ) ;
testsect [ 2 ] = 2 * ( blocksize > > 9 ) ;
testcount = 3 ;
}
for ( i = 0 ; i < testcount ; + + i ) {
data = read_part_sector ( state , testsect [ i ] , & sect ) ;
if ( data = = NULL )
continue ;
memcpy ( label , data , sizeof ( * label ) ) ;
memcpy ( temp , data , 4 ) ;
temp [ 4 ] = 0 ;
EBCASC ( temp , 4 ) ;
put_dev_sector ( sect ) ;
if ( ! strcmp ( temp , " VOL1 " ) | |
! strcmp ( temp , " LNX1 " ) | |
! strcmp ( temp , " CMS1 " ) ) {
if ( ! strcmp ( temp , " VOL1 " ) ) {
strncpy ( type , label - > vol . vollbl , 4 ) ;
strncpy ( name , label - > vol . volid , 6 ) ;
} else {
strncpy ( type , label - > lnx . vollbl , 4 ) ;
strncpy ( name , label - > lnx . volid , 6 ) ;
}
EBCASC ( type , 4 ) ;
EBCASC ( name , 6 ) ;
* labelsect = testsect [ i ] ;
found = 1 ;
break ;
}
}
if ( ! found )
memset ( label , 0 , sizeof ( * label ) ) ;
return found ;
}
static int find_vol1_partitions ( struct parsed_partitions * state ,
struct hd_geometry * geo ,
int blocksize ,
char name [ ] ,
union label_t * label )
{
sector_t blk ;
int counter ;
char tmp [ 64 ] ;
Sector sect ;
unsigned char * data ;
loff_t offset , size ;
struct vtoc_format1_label f1 ;
int secperblk ;
snprintf ( tmp , sizeof ( tmp ) , " VOL1/%8s: " , name ) ;
strlcat ( state - > pp_buf , tmp , PAGE_SIZE ) ;
/*
* get start of VTOC from the disk label and then search for format1
* and format8 labels
*/
secperblk = blocksize > > 9 ;
blk = cchhb2blk ( & label - > vol . vtoc , geo ) + 1 ;
counter = 0 ;
data = read_part_sector ( state , blk * secperblk , & sect ) ;
while ( data ! = NULL ) {
memcpy ( & f1 , data , sizeof ( struct vtoc_format1_label ) ) ;
put_dev_sector ( sect ) ;
/* skip FMT4 / FMT5 / FMT7 labels */
if ( f1 . DS1FMTID = = _ascebc [ ' 4 ' ]
| | f1 . DS1FMTID = = _ascebc [ ' 5 ' ]
| | f1 . DS1FMTID = = _ascebc [ ' 7 ' ]
| | f1 . DS1FMTID = = _ascebc [ ' 9 ' ] ) {
blk + + ;
data = read_part_sector ( state , blk * secperblk , & sect ) ;
continue ;
}
/* only FMT1 and 8 labels valid at this point */
if ( f1 . DS1FMTID ! = _ascebc [ ' 1 ' ] & &
f1 . DS1FMTID ! = _ascebc [ ' 8 ' ] )
break ;
/* OK, we got valid partition data */
offset = cchh2blk ( & f1 . DS1EXT1 . llimit , geo ) ;
size = cchh2blk ( & f1 . DS1EXT1 . ulimit , geo ) -
offset + geo - > sectors ;
offset * = secperblk ;
size * = secperblk ;
if ( counter > = state - > limit )
break ;
put_partition ( state , counter + 1 , offset , size ) ;
counter + + ;
blk + + ;
data = read_part_sector ( state , blk * secperblk , & sect ) ;
}
strlcat ( state - > pp_buf , " \n " , PAGE_SIZE ) ;
if ( ! data )
return - 1 ;
return 1 ;
}
static int find_lnx1_partitions ( struct parsed_partitions * state ,
struct hd_geometry * geo ,
int blocksize ,
char name [ ] ,
union label_t * label ,
sector_t labelsect ,
loff_t i_size ,
dasd_information2_t * info )
{
loff_t offset , geo_size , size ;
char tmp [ 64 ] ;
int secperblk ;
snprintf ( tmp , sizeof ( tmp ) , " LNX1/%8s: " , name ) ;
strlcat ( state - > pp_buf , tmp , PAGE_SIZE ) ;
secperblk = blocksize > > 9 ;
if ( label - > lnx . ldl_version = = 0xf2 ) {
size = label - > lnx . formatted_blocks * secperblk ;
} else {
/*
* Formated w / o large volume support . If the sanity check
* ' size based on geo = = size based on i_size ' is true , then
* we can safely assume that we know the formatted size of
* the disk , otherwise we need additional information
* that we can only get from a real DASD device .
*/
geo_size = geo - > cylinders * geo - > heads
* geo - > sectors * secperblk ;
size = i_size > > 9 ;
if ( size ! = geo_size ) {
if ( ! info ) {
strlcat ( state - > pp_buf , " \n " , PAGE_SIZE ) ;
return 1 ;
}
if ( ! strcmp ( info - > type , " ECKD " ) )
if ( geo_size < size )
size = geo_size ;
/* else keep size based on i_size */
}
}
/* first and only partition starts in the first block after the label */
offset = labelsect + secperblk ;
put_partition ( state , 1 , offset , size - offset ) ;
strlcat ( state - > pp_buf , " \n " , PAGE_SIZE ) ;
return 1 ;
}
static int find_cms1_partitions ( struct parsed_partitions * state ,
struct hd_geometry * geo ,
int blocksize ,
char name [ ] ,
union label_t * label ,
sector_t labelsect )
{
loff_t offset , size ;
char tmp [ 64 ] ;
int secperblk ;
/*
* VM style CMS1 labeled disk
*/
blocksize = label - > cms . block_size ;
secperblk = blocksize > > 9 ;
if ( label - > cms . disk_offset ! = 0 ) {
snprintf ( tmp , sizeof ( tmp ) , " CMS1/%8s(MDSK): " , name ) ;
strlcat ( state - > pp_buf , tmp , PAGE_SIZE ) ;
/* disk is reserved minidisk */
offset = label - > cms . disk_offset * secperblk ;
size = ( label - > cms . block_count - 1 ) * secperblk ;
} else {
snprintf ( tmp , sizeof ( tmp ) , " CMS1/%8s: " , name ) ;
strlcat ( state - > pp_buf , tmp , PAGE_SIZE ) ;
/*
* Special case for FBA devices :
* If an FBA device is CMS formatted with blocksize > 512 byte
* and the DIAG discipline is used , then the CMS label is found
* in sector 1 instead of block 1. However , the partition is
* still supposed to start in block 2.
*/
if ( labelsect = = 1 )
offset = 2 * secperblk ;
else
offset = labelsect + secperblk ;
size = label - > cms . block_count * secperblk ;
}
put_partition ( state , 1 , offset , size - offset ) ;
strlcat ( state - > pp_buf , " \n " , PAGE_SIZE ) ;
return 1 ;
}
2005-04-16 15:20:36 -07:00
/*
2012-08-31 10:52:13 +02:00
* This is the main function , called by check . c
2005-04-16 15:20:36 -07:00
*/
2010-05-15 20:09:30 +02:00
int ibm_partition ( struct parsed_partitions * state )
2005-04-16 15:20:36 -07:00
{
2010-05-15 20:09:30 +02:00
struct block_device * bdev = state - > bdev ;
2009-03-26 15:23:47 +01:00
int blocksize , res ;
2012-08-31 10:52:13 +02:00
loff_t i_size , offset , size ;
2007-07-10 11:24:11 +02:00
dasd_information2_t * info ;
2005-04-16 15:20:36 -07:00
struct hd_geometry * geo ;
char type [ 5 ] = { 0 , } ;
char name [ 7 ] = { 0 , } ;
2010-07-19 09:22:35 +02:00
sector_t labelsect ;
2012-08-31 10:52:13 +02:00
union label_t * label ;
2005-04-16 15:20:36 -07:00
2006-12-06 20:35:16 -08:00
res = 0 ;
2009-05-22 17:17:49 -04:00
blocksize = bdev_logical_block_size ( bdev ) ;
2006-03-07 21:55:39 -08:00
if ( blocksize < = 0 )
2006-12-06 20:35:16 -08:00
goto out_exit ;
2006-03-07 21:55:39 -08:00
i_size = i_size_read ( bdev - > bd_inode ) ;
if ( i_size = = 0 )
2006-12-06 20:35:16 -08:00
goto out_exit ;
2007-07-10 11:24:11 +02:00
info = kmalloc ( sizeof ( dasd_information2_t ) , GFP_KERNEL ) ;
if ( info = = NULL )
2006-12-06 20:35:16 -08:00
goto out_exit ;
2007-07-10 11:24:11 +02:00
geo = kmalloc ( sizeof ( struct hd_geometry ) , GFP_KERNEL ) ;
if ( geo = = NULL )
2005-04-16 15:20:36 -07:00
goto out_nogeo ;
2007-07-10 11:24:11 +02:00
label = kmalloc ( sizeof ( union label_t ) , GFP_KERNEL ) ;
if ( label = = NULL )
2006-01-06 00:19:09 -08:00
goto out_nolab ;
2012-08-31 10:52:13 +02:00
if ( ioctl_by_bdev ( bdev , HDIO_GETGEO , ( unsigned long ) geo ) ! = 0 )
2006-12-06 20:35:16 -08:00
goto out_freeall ;
2012-08-31 10:52:13 +02:00
if ( ioctl_by_bdev ( bdev , BIODASDINFO2 , ( unsigned long ) info ) ! = 0 ) {
kfree ( info ) ;
info = NULL ;
2009-03-26 15:23:47 +01:00
}
2006-12-06 20:35:16 -08:00
2012-08-31 10:52:13 +02:00
if ( find_label ( state , info , geo , blocksize , & labelsect , name , type ,
label ) ) {
if ( ! strncmp ( type , " VOL1 " , 4 ) ) {
res = find_vol1_partitions ( state , geo , blocksize , name ,
label ) ;
} else if ( ! strncmp ( type , " LNX1 " , 4 ) ) {
res = find_lnx1_partitions ( state , geo , blocksize , name ,
label , labelsect , i_size ,
info ) ;
} else if ( ! strncmp ( type , " CMS1 " , 4 ) ) {
res = find_cms1_partitions ( state , geo , blocksize , name ,
label , labelsect ) ;
2009-03-26 15:23:47 +01:00
}
2012-08-31 10:52:13 +02:00
} else if ( info ) {
2005-04-16 15:20:36 -07:00
/*
2012-08-31 10:52:13 +02:00
* ugly but needed for backward compatibility :
* If the block device is a DASD ( i . e . BIODASDINFO2 works ) ,
* then we claim it in any case , even though it has no valid
* label . If it has the LDL format , then we simply define a
* partition as if it had an LNX1 label .
2005-04-16 15:20:36 -07:00
*/
2012-08-31 10:52:13 +02:00
res = 1 ;
if ( info - > format = = DASD_FORMAT_LDL ) {
strlcat ( state - > pp_buf , " (nonl) " , PAGE_SIZE ) ;
size = i_size > > 9 ;
offset = ( info - > label_block + 1 ) * ( blocksize > > 9 ) ;
put_partition ( state , 1 , offset , size - offset ) ;
strlcat ( state - > pp_buf , " \n " , PAGE_SIZE ) ;
}
} else
res = 0 ;
2006-03-24 03:15:23 -08:00
2006-12-06 20:35:16 -08:00
out_freeall :
2006-01-06 00:19:09 -08:00
kfree ( label ) ;
out_nolab :
2005-04-16 15:20:36 -07:00
kfree ( geo ) ;
out_nogeo :
kfree ( info ) ;
2006-12-06 20:35:16 -08:00
out_exit :
return res ;
2005-04-16 15:20:36 -07:00
}