2006-06-29 14:58:12 +02:00
/*
2005-04-16 15:20:36 -07:00
* Author ( s ) . . . . . . : Holger Smolinski < Holger . Smolinski @ de . ibm . com >
2006-06-29 14:58:12 +02:00
* Horst Hummel < Horst . Hummel @ de . ibm . com >
2005-04-16 15:20:36 -07:00
* Carsten Otte < Cotte @ de . ibm . com >
* Martin Schwidefsky < schwidefsky @ de . ibm . com >
* Bugreports . to . . : < Linux390 @ de . ibm . com >
2009-06-16 10:30:25 +02:00
* Copyright IBM Corp . 1999 , 2009
2008-10-10 21:33:25 +02:00
* EMC Symmetrix ioctl Copyright EMC Corporation , 2008
* Author . . . . . . . . . : Nigel Hislop < hislop_nigel @ emc . com >
2005-04-16 15:20:36 -07:00
*/
2009-09-11 10:28:30 +02:00
# define KMSG_COMPONENT "dasd-eckd"
2009-03-26 15:23:49 +01:00
2005-04-16 15:20:36 -07:00
# include <linux/stddef.h>
# include <linux/kernel.h>
# include <linux/slab.h>
# include <linux/hdreg.h> /* HDIO_GETGEO */
# include <linux/bio.h>
# include <linux/module.h>
2012-02-27 10:01:52 +01:00
# include <linux/compat.h>
2005-04-16 15:20:36 -07:00
# include <linux/init.h>
2012-08-28 16:43:36 +02:00
# include <asm/css_chars.h>
2005-04-16 15:20:36 -07:00
# include <asm/debug.h>
# include <asm/idals.h>
# include <asm/ebcdic.h>
# include <asm/io.h>
# include <asm/uaccess.h>
2006-06-29 15:08:18 +02:00
# include <asm/cio.h>
2005-04-16 15:20:36 -07:00
# include <asm/ccwdev.h>
2009-03-26 15:23:48 +01:00
# include <asm/itcw.h>
2014-10-01 14:39:47 +02:00
# include <asm/schid.h>
# include <asm/chpid.h>
2005-04-16 15:20:36 -07:00
# include "dasd_int.h"
# include "dasd_eckd.h"
# ifdef PRINTK_HEADER
# undef PRINTK_HEADER
# endif /* PRINTK_HEADER */
# define PRINTK_HEADER "dasd(eckd):"
# define ECKD_C0(i) (i->home_bytes)
# define ECKD_F(i) (i->formula)
# define ECKD_F1(i) (ECKD_F(i)==0x01?(i->factors.f_0x01.f1):\
( i - > factors . f_0x02 . f1 ) )
# define ECKD_F2(i) (ECKD_F(i)==0x01?(i->factors.f_0x01.f2):\
( i - > factors . f_0x02 . f2 ) )
# define ECKD_F3(i) (ECKD_F(i)==0x01?(i->factors.f_0x01.f3):\
( i - > factors . f_0x02 . f3 ) )
# define ECKD_F4(i) (ECKD_F(i)==0x02?(i->factors.f_0x02.f4):0)
# define ECKD_F5(i) (ECKD_F(i)==0x02?(i->factors.f_0x02.f5):0)
# define ECKD_F6(i) (i->factor6)
# define ECKD_F7(i) (i->factor7)
# define ECKD_F8(i) (i->factor8)
2011-01-05 12:48:06 +01:00
/*
* raw track access always map to 64 k in memory
* so it maps to 16 blocks of 4 k per track
*/
# define DASD_RAW_BLOCK_PER_TRACK 16
# define DASD_RAW_BLOCKSIZE 4096
/* 64k are 128 x 512 byte sectors */
# define DASD_RAW_SECTORS_PER_TRACK 128
2005-04-16 15:20:36 -07:00
MODULE_LICENSE ( " GPL " ) ;
static struct dasd_discipline dasd_eckd_discipline ;
/* The ccw bus type uses this table to find devices that it sends to
* dasd_eckd_probe */
static struct ccw_device_id dasd_eckd_ids [ ] = {
2006-07-12 16:41:55 +02:00
{ CCW_DEVICE_DEVTYPE ( 0x3990 , 0 , 0x3390 , 0 ) , . driver_info = 0x1 } ,
{ CCW_DEVICE_DEVTYPE ( 0x2105 , 0 , 0x3390 , 0 ) , . driver_info = 0x2 } ,
2011-02-17 13:13:55 +01:00
{ CCW_DEVICE_DEVTYPE ( 0x3880 , 0 , 0x3380 , 0 ) , . driver_info = 0x3 } ,
2006-07-12 16:41:55 +02:00
{ CCW_DEVICE_DEVTYPE ( 0x3990 , 0 , 0x3380 , 0 ) , . driver_info = 0x4 } ,
{ CCW_DEVICE_DEVTYPE ( 0x2105 , 0 , 0x3380 , 0 ) , . driver_info = 0x5 } ,
{ CCW_DEVICE_DEVTYPE ( 0x9343 , 0 , 0x9345 , 0 ) , . driver_info = 0x6 } ,
{ CCW_DEVICE_DEVTYPE ( 0x2107 , 0 , 0x3390 , 0 ) , . driver_info = 0x7 } ,
{ CCW_DEVICE_DEVTYPE ( 0x2107 , 0 , 0x3380 , 0 ) , . driver_info = 0x8 } ,
{ CCW_DEVICE_DEVTYPE ( 0x1750 , 0 , 0x3390 , 0 ) , . driver_info = 0x9 } ,
{ CCW_DEVICE_DEVTYPE ( 0x1750 , 0 , 0x3380 , 0 ) , . driver_info = 0xa } ,
2005-04-16 15:20:36 -07:00
{ /* end of list */ } ,
} ;
MODULE_DEVICE_TABLE ( ccw , dasd_eckd_ids ) ;
static struct ccw_driver dasd_eckd_driver ; /* see below */
2013-08-16 15:57:32 +02:00
static void * rawpadpage ;
2009-12-07 12:51:51 +01:00
# define INIT_CQR_OK 0
# define INIT_CQR_UNFORMATTED 1
# define INIT_CQR_ERROR 2
2010-08-09 18:12:59 +02:00
/* emergency request for reserve/release */
static struct {
struct dasd_ccw_req cqr ;
struct ccw1 ccw ;
char data [ 32 ] ;
} * dasd_reserve_req ;
static DEFINE_MUTEX ( dasd_reserve_mutex ) ;
2011-01-05 12:48:03 +01:00
/* definitions for the path verification worker */
struct path_verification_work_data {
struct work_struct worker ;
struct dasd_device * device ;
struct dasd_ccw_req cqr ;
struct ccw1 ccw ;
__u8 rcd_buffer [ DASD_ECKD_RCD_DATA_SIZE ] ;
int isglobal ;
__u8 tbvpm ;
} ;
static struct path_verification_work_data * path_verification_worker ;
static DEFINE_MUTEX ( dasd_path_verification_mutex ) ;
2009-12-07 12:51:51 +01:00
2014-10-01 14:39:47 +02:00
struct check_attention_work_data {
struct work_struct worker ;
struct dasd_device * device ;
__u8 lpum ;
} ;
2005-04-16 15:20:36 -07:00
/* initial attempt at a probe function. this can be simplified once
* the other detection code is gone */
static int
dasd_eckd_probe ( struct ccw_device * cdev )
{
int ret ;
2006-06-29 15:08:18 +02:00
/* set ECKD specific ccw-device options */
2009-12-07 12:51:30 +01:00
ret = ccw_device_set_options ( cdev , CCWDEV_ALLOW_FORCE |
CCWDEV_DO_PATHGROUP | CCWDEV_DO_MULTIPATH ) ;
2006-06-29 15:08:18 +02:00
if ( ret ) {
2009-12-07 12:51:52 +01:00
DBF_EVENT_DEVID ( DBF_WARNING , cdev , " %s " ,
" dasd_eckd_probe: could not set "
" ccw-device options " ) ;
2005-04-16 15:20:36 -07:00
return ret ;
2006-06-29 15:08:18 +02:00
}
ret = dasd_generic_probe ( cdev , & dasd_eckd_discipline ) ;
return ret ;
2005-04-16 15:20:36 -07:00
}
static int
dasd_eckd_set_online ( struct ccw_device * cdev )
{
2006-06-29 15:08:18 +02:00
return dasd_generic_set_online ( cdev , & dasd_eckd_discipline ) ;
2005-04-16 15:20:36 -07:00
}
static const int sizes_trk0 [ ] = { 28 , 148 , 84 } ;
# define LABEL_SIZE 140
2012-09-20 18:37:36 +02:00
/* head and record addresses of count_area read in analysis ccw */
static const int count_area_head [ ] = { 0 , 0 , 0 , 0 , 2 } ;
static const int count_area_rec [ ] = { 1 , 2 , 3 , 4 , 1 } ;
2005-04-16 15:20:36 -07:00
static inline unsigned int
round_up_multiple ( unsigned int no , unsigned int mult )
{
int rem = no % mult ;
return ( rem ? no - rem + mult : no ) ;
}
static inline unsigned int
ceil_quot ( unsigned int d1 , unsigned int d2 )
{
return ( d1 + ( d2 - 1 ) ) / d2 ;
}
2007-02-05 21:18:53 +01:00
static unsigned int
2005-04-16 15:20:36 -07:00
recs_per_track ( struct dasd_eckd_characteristics * rdc ,
unsigned int kl , unsigned int dl )
{
int dn , kn ;
switch ( rdc - > dev_type ) {
case 0x3380 :
if ( kl )
return 1499 / ( 15 + 7 + ceil_quot ( kl + 12 , 32 ) +
ceil_quot ( dl + 12 , 32 ) ) ;
else
return 1499 / ( 15 + ceil_quot ( dl + 12 , 32 ) ) ;
case 0x3390 :
dn = ceil_quot ( dl + 6 , 232 ) + 1 ;
if ( kl ) {
kn = ceil_quot ( kl + 6 , 232 ) + 1 ;
return 1729 / ( 10 + 9 + ceil_quot ( kl + 6 * kn , 34 ) +
9 + ceil_quot ( dl + 6 * dn , 34 ) ) ;
} else
return 1729 / ( 10 + 9 + ceil_quot ( dl + 6 * dn , 34 ) ) ;
case 0x9345 :
dn = ceil_quot ( dl + 6 , 232 ) + 1 ;
if ( kl ) {
kn = ceil_quot ( kl + 6 , 232 ) + 1 ;
return 1420 / ( 18 + 7 + ceil_quot ( kl + 6 * kn , 34 ) +
ceil_quot ( dl + 6 * dn , 34 ) ) ;
} else
return 1420 / ( 18 + 7 + ceil_quot ( dl + 6 * dn , 34 ) ) ;
}
return 0 ;
}
2009-03-26 15:23:47 +01:00
static void set_ch_t ( struct ch_t * geo , __u32 cyl , __u8 head )
{
geo - > cyl = ( __u16 ) cyl ;
geo - > head = cyl > > 16 ;
geo - > head < < = 4 ;
geo - > head | = head ;
}
2007-02-05 21:18:53 +01:00
static int
2005-04-16 15:20:36 -07:00
check_XRC ( struct ccw1 * de_ccw ,
struct DE_eckd_data * data ,
struct dasd_device * device )
{
struct dasd_eckd_private * private ;
2007-02-05 21:18:19 +01:00
int rc ;
2005-04-16 15:20:36 -07:00
private = ( struct dasd_eckd_private * ) device - > private ;
2007-02-05 21:18:19 +01:00
if ( ! private - > rdc_data . facilities . XRC_supported )
return 0 ;
2005-04-16 15:20:36 -07:00
/* switch on System Time Stamp - needed for XRC Support */
2007-02-05 21:18:19 +01:00
data - > ga_extended | = 0x08 ; /* switch on 'Time Stamp Valid' */
data - > ga_extended | = 0x02 ; /* switch on 'Extended Parameter' */
2005-04-16 15:20:36 -07:00
2007-02-05 21:18:19 +01:00
rc = get_sync_clock ( & data - > ep_sys_time ) ;
/* Ignore return code if sync clock is switched off. */
2012-09-06 15:13:34 +02:00
if ( rc = = - EOPNOTSUPP | | rc = = - EACCES )
2007-02-05 21:18:19 +01:00
rc = 0 ;
2005-04-16 15:20:36 -07:00
2008-01-26 14:11:23 +01:00
de_ccw - > count = sizeof ( struct DE_eckd_data ) ;
2007-02-05 21:18:19 +01:00
de_ccw - > flags | = CCW_FLAG_SLI ;
return rc ;
}
2005-04-16 15:20:36 -07:00
2007-02-05 21:18:53 +01:00
static int
2009-03-26 15:23:47 +01:00
define_extent ( struct ccw1 * ccw , struct DE_eckd_data * data , unsigned int trk ,
unsigned int totrk , int cmd , struct dasd_device * device )
2005-04-16 15:20:36 -07:00
{
struct dasd_eckd_private * private ;
2009-03-26 15:23:47 +01:00
u32 begcyl , endcyl ;
u16 heads , beghead , endhead ;
2007-02-05 21:18:19 +01:00
int rc = 0 ;
2005-04-16 15:20:36 -07:00
private = ( struct dasd_eckd_private * ) device - > private ;
ccw - > cmd_code = DASD_ECKD_CCW_DEFINE_EXTENT ;
ccw - > flags = 0 ;
ccw - > count = 16 ;
ccw - > cda = ( __u32 ) __pa ( data ) ;
2008-01-26 14:11:23 +01:00
memset ( data , 0 , sizeof ( struct DE_eckd_data ) ) ;
2005-04-16 15:20:36 -07:00
switch ( cmd ) {
case DASD_ECKD_CCW_READ_HOME_ADDRESS :
case DASD_ECKD_CCW_READ_RECORD_ZERO :
case DASD_ECKD_CCW_READ :
case DASD_ECKD_CCW_READ_MT :
case DASD_ECKD_CCW_READ_CKD :
case DASD_ECKD_CCW_READ_CKD_MT :
case DASD_ECKD_CCW_READ_KD :
case DASD_ECKD_CCW_READ_KD_MT :
case DASD_ECKD_CCW_READ_COUNT :
data - > mask . perm = 0x1 ;
data - > attributes . operation = private - > attrib . operation ;
break ;
case DASD_ECKD_CCW_WRITE :
case DASD_ECKD_CCW_WRITE_MT :
case DASD_ECKD_CCW_WRITE_KD :
case DASD_ECKD_CCW_WRITE_KD_MT :
data - > mask . perm = 0x02 ;
data - > attributes . operation = private - > attrib . operation ;
2007-02-05 21:18:19 +01:00
rc = check_XRC ( ccw , data , device ) ;
2005-04-16 15:20:36 -07:00
break ;
case DASD_ECKD_CCW_WRITE_CKD :
case DASD_ECKD_CCW_WRITE_CKD_MT :
data - > attributes . operation = DASD_BYPASS_CACHE ;
2007-02-05 21:18:19 +01:00
rc = check_XRC ( ccw , data , device ) ;
2005-04-16 15:20:36 -07:00
break ;
case DASD_ECKD_CCW_ERASE :
case DASD_ECKD_CCW_WRITE_HOME_ADDRESS :
case DASD_ECKD_CCW_WRITE_RECORD_ZERO :
data - > mask . perm = 0x3 ;
data - > mask . auth = 0x1 ;
data - > attributes . operation = DASD_BYPASS_CACHE ;
2007-02-05 21:18:19 +01:00
rc = check_XRC ( ccw , data , device ) ;
2005-04-16 15:20:36 -07:00
break ;
default :
2009-03-26 15:23:49 +01:00
dev_err ( & device - > cdev - > dev ,
" 0x%x is not a known command \n " , cmd ) ;
2005-04-16 15:20:36 -07:00
break ;
}
data - > attributes . mode = 0x3 ; /* ECKD */
if ( ( private - > rdc_data . cu_type = = 0x2105 | |
private - > rdc_data . cu_type = = 0x2107 | |
private - > rdc_data . cu_type = = 0x1750 )
& & ! ( private - > uses_cdl & & trk < 2 ) )
data - > ga_extended | = 0x40 ; /* Regular Data Format Mode */
2009-03-26 15:23:47 +01:00
heads = private - > rdc_data . trk_per_cyl ;
begcyl = trk / heads ;
beghead = trk % heads ;
endcyl = totrk / heads ;
endhead = totrk % heads ;
2005-04-16 15:20:36 -07:00
/* check for sequential prestage - enhance cylinder range */
if ( data - > attributes . operation = = DASD_SEQ_PRESTAGE | |
data - > attributes . operation = = DASD_SEQ_ACCESS ) {
2006-06-29 14:58:12 +02:00
2009-03-26 15:23:47 +01:00
if ( endcyl + private - > attrib . nr_cyl < private - > real_cyl )
endcyl + = private - > attrib . nr_cyl ;
2005-04-16 15:20:36 -07:00
else
2009-03-26 15:23:47 +01:00
endcyl = ( private - > real_cyl - 1 ) ;
2005-04-16 15:20:36 -07:00
}
2009-03-26 15:23:47 +01:00
set_ch_t ( & data - > beg_ext , begcyl , beghead ) ;
set_ch_t ( & data - > end_ext , endcyl , endhead ) ;
2007-02-05 21:18:19 +01:00
return rc ;
2005-04-16 15:20:36 -07:00
}
2008-01-26 14:11:23 +01:00
static int check_XRC_on_prefix ( struct PFX_eckd_data * pfxdata ,
struct dasd_device * device )
{
struct dasd_eckd_private * private ;
int rc ;
private = ( struct dasd_eckd_private * ) device - > private ;
if ( ! private - > rdc_data . facilities . XRC_supported )
return 0 ;
/* switch on System Time Stamp - needed for XRC Support */
2009-03-26 15:23:48 +01:00
pfxdata - > define_extent . ga_extended | = 0x08 ; /* 'Time Stamp Valid' */
pfxdata - > define_extent . ga_extended | = 0x02 ; /* 'Extended Parameter' */
2008-01-26 14:11:23 +01:00
pfxdata - > validity . time_stamp = 1 ; /* 'Time Stamp Valid' */
2009-03-26 15:23:48 +01:00
rc = get_sync_clock ( & pfxdata - > define_extent . ep_sys_time ) ;
2008-01-26 14:11:23 +01:00
/* Ignore return code if sync clock is switched off. */
2012-09-06 15:13:34 +02:00
if ( rc = = - EOPNOTSUPP | | rc = = - EACCES )
2008-01-26 14:11:23 +01:00
rc = 0 ;
return rc ;
}
2009-03-26 15:23:48 +01:00
static void fill_LRE_data ( struct LRE_eckd_data * data , unsigned int trk ,
unsigned int rec_on_trk , int count , int cmd ,
struct dasd_device * device , unsigned int reclen ,
unsigned int tlf )
{
struct dasd_eckd_private * private ;
int sector ;
int dn , d ;
private = ( struct dasd_eckd_private * ) device - > private ;
memset ( data , 0 , sizeof ( * data ) ) ;
sector = 0 ;
if ( rec_on_trk ) {
switch ( private - > rdc_data . dev_type ) {
case 0x3390 :
dn = ceil_quot ( reclen + 6 , 232 ) ;
d = 9 + ceil_quot ( reclen + 6 * ( dn + 1 ) , 34 ) ;
sector = ( 49 + ( rec_on_trk - 1 ) * ( 10 + d ) ) / 8 ;
break ;
case 0x3380 :
d = 7 + ceil_quot ( reclen + 12 , 32 ) ;
sector = ( 39 + ( rec_on_trk - 1 ) * ( 8 + d ) ) / 7 ;
break ;
}
}
data - > sector = sector ;
/* note: meaning of count depends on the operation
* for record based I / O it ' s the number of records , but for
* track based I / O it ' s the number of tracks
*/
data - > count = count ;
switch ( cmd ) {
case DASD_ECKD_CCW_WRITE_HOME_ADDRESS :
data - > operation . orientation = 0x3 ;
data - > operation . operation = 0x03 ;
break ;
case DASD_ECKD_CCW_READ_HOME_ADDRESS :
data - > operation . orientation = 0x3 ;
data - > operation . operation = 0x16 ;
break ;
case DASD_ECKD_CCW_WRITE_RECORD_ZERO :
data - > operation . orientation = 0x1 ;
data - > operation . operation = 0x03 ;
data - > count + + ;
break ;
case DASD_ECKD_CCW_READ_RECORD_ZERO :
data - > operation . orientation = 0x3 ;
data - > operation . operation = 0x16 ;
data - > count + + ;
break ;
case DASD_ECKD_CCW_WRITE :
case DASD_ECKD_CCW_WRITE_MT :
case DASD_ECKD_CCW_WRITE_KD :
case DASD_ECKD_CCW_WRITE_KD_MT :
data - > auxiliary . length_valid = 0x1 ;
data - > length = reclen ;
data - > operation . operation = 0x01 ;
break ;
case DASD_ECKD_CCW_WRITE_CKD :
case DASD_ECKD_CCW_WRITE_CKD_MT :
data - > auxiliary . length_valid = 0x1 ;
data - > length = reclen ;
data - > operation . operation = 0x03 ;
break ;
2011-01-05 12:48:06 +01:00
case DASD_ECKD_CCW_WRITE_FULL_TRACK :
data - > operation . orientation = 0x0 ;
data - > operation . operation = 0x3F ;
data - > extended_operation = 0x11 ;
data - > length = 0 ;
data - > extended_parameter_length = 0x02 ;
if ( data - > count > 8 ) {
data - > extended_parameter [ 0 ] = 0xFF ;
data - > extended_parameter [ 1 ] = 0xFF ;
data - > extended_parameter [ 1 ] < < = ( 16 - count ) ;
} else {
data - > extended_parameter [ 0 ] = 0xFF ;
data - > extended_parameter [ 0 ] < < = ( 8 - count ) ;
data - > extended_parameter [ 1 ] = 0x00 ;
}
data - > sector = 0xFF ;
break ;
2009-03-26 15:23:48 +01:00
case DASD_ECKD_CCW_WRITE_TRACK_DATA :
data - > auxiliary . length_valid = 0x1 ;
data - > length = reclen ; /* not tlf, as one might think */
data - > operation . operation = 0x3F ;
data - > extended_operation = 0x23 ;
break ;
case DASD_ECKD_CCW_READ :
case DASD_ECKD_CCW_READ_MT :
case DASD_ECKD_CCW_READ_KD :
case DASD_ECKD_CCW_READ_KD_MT :
data - > auxiliary . length_valid = 0x1 ;
data - > length = reclen ;
data - > operation . operation = 0x06 ;
break ;
case DASD_ECKD_CCW_READ_CKD :
case DASD_ECKD_CCW_READ_CKD_MT :
data - > auxiliary . length_valid = 0x1 ;
data - > length = reclen ;
data - > operation . operation = 0x16 ;
break ;
case DASD_ECKD_CCW_READ_COUNT :
data - > operation . operation = 0x06 ;
break ;
2011-01-05 12:48:06 +01:00
case DASD_ECKD_CCW_READ_TRACK :
data - > operation . orientation = 0x1 ;
data - > operation . operation = 0x0C ;
data - > extended_parameter_length = 0 ;
data - > sector = 0xFF ;
break ;
2009-03-26 15:23:48 +01:00
case DASD_ECKD_CCW_READ_TRACK_DATA :
data - > auxiliary . length_valid = 0x1 ;
data - > length = tlf ;
data - > operation . operation = 0x0C ;
break ;
case DASD_ECKD_CCW_ERASE :
data - > length = reclen ;
data - > auxiliary . length_valid = 0x1 ;
data - > operation . operation = 0x0b ;
break ;
default :
DBF_DEV_EVENT ( DBF_ERR , device ,
" fill LRE unknown opcode 0x%x " , cmd ) ;
BUG ( ) ;
}
set_ch_t ( & data - > seek_addr ,
trk / private - > rdc_data . trk_per_cyl ,
trk % private - > rdc_data . trk_per_cyl ) ;
data - > search_arg . cyl = data - > seek_addr . cyl ;
data - > search_arg . head = data - > seek_addr . head ;
data - > search_arg . record = rec_on_trk ;
}
static int prefix_LRE ( struct ccw1 * ccw , struct PFX_eckd_data * pfxdata ,
unsigned int trk , unsigned int totrk , int cmd ,
struct dasd_device * basedev , struct dasd_device * startdev ,
unsigned char format , unsigned int rec_on_trk , int count ,
unsigned int blksize , unsigned int tlf )
2008-01-26 14:11:23 +01:00
{
struct dasd_eckd_private * basepriv , * startpriv ;
2009-03-26 15:23:48 +01:00
struct DE_eckd_data * dedata ;
struct LRE_eckd_data * lredata ;
2009-03-26 15:23:47 +01:00
u32 begcyl , endcyl ;
u16 heads , beghead , endhead ;
2008-01-26 14:11:23 +01:00
int rc = 0 ;
basepriv = ( struct dasd_eckd_private * ) basedev - > private ;
startpriv = ( struct dasd_eckd_private * ) startdev - > private ;
2009-03-26 15:23:48 +01:00
dedata = & pfxdata - > define_extent ;
lredata = & pfxdata - > locate_record ;
2008-01-26 14:11:23 +01:00
ccw - > cmd_code = DASD_ECKD_CCW_PFX ;
ccw - > flags = 0 ;
2011-01-05 12:48:06 +01:00
if ( cmd = = DASD_ECKD_CCW_WRITE_FULL_TRACK ) {
ccw - > count = sizeof ( * pfxdata ) + 2 ;
ccw - > cda = ( __u32 ) __pa ( pfxdata ) ;
memset ( pfxdata , 0 , sizeof ( * pfxdata ) + 2 ) ;
} else {
ccw - > count = sizeof ( * pfxdata ) ;
ccw - > cda = ( __u32 ) __pa ( pfxdata ) ;
memset ( pfxdata , 0 , sizeof ( * pfxdata ) ) ;
}
2008-01-26 14:11:23 +01:00
/* prefix data */
2009-03-26 15:23:48 +01:00
if ( format > 1 ) {
DBF_DEV_EVENT ( DBF_ERR , basedev ,
" PFX LRE unknown format 0x%x " , format ) ;
BUG ( ) ;
return - EINVAL ;
}
pfxdata - > format = format ;
2008-08-01 16:39:09 +02:00
pfxdata - > base_address = basepriv - > ned - > unit_addr ;
pfxdata - > base_lss = basepriv - > ned - > ID ;
2009-03-26 15:23:48 +01:00
pfxdata - > validity . define_extent = 1 ;
2008-01-26 14:11:23 +01:00
/* private uid is kept up to date, conf_data may be outdated */
if ( startpriv - > uid . type ! = UA_BASE_DEVICE ) {
pfxdata - > validity . verify_base = 1 ;
if ( startpriv - > uid . type = = UA_HYPER_PAV_ALIAS )
pfxdata - > validity . hyper_pav = 1 ;
}
/* define extend data (mostly)*/
switch ( cmd ) {
case DASD_ECKD_CCW_READ_HOME_ADDRESS :
case DASD_ECKD_CCW_READ_RECORD_ZERO :
case DASD_ECKD_CCW_READ :
case DASD_ECKD_CCW_READ_MT :
case DASD_ECKD_CCW_READ_CKD :
case DASD_ECKD_CCW_READ_CKD_MT :
case DASD_ECKD_CCW_READ_KD :
case DASD_ECKD_CCW_READ_KD_MT :
case DASD_ECKD_CCW_READ_COUNT :
2009-03-26 15:23:48 +01:00
dedata - > mask . perm = 0x1 ;
dedata - > attributes . operation = basepriv - > attrib . operation ;
break ;
2011-01-05 12:48:06 +01:00
case DASD_ECKD_CCW_READ_TRACK :
2009-03-26 15:23:48 +01:00
case DASD_ECKD_CCW_READ_TRACK_DATA :
dedata - > mask . perm = 0x1 ;
dedata - > attributes . operation = basepriv - > attrib . operation ;
dedata - > blk_size = 0 ;
2008-01-26 14:11:23 +01:00
break ;
case DASD_ECKD_CCW_WRITE :
case DASD_ECKD_CCW_WRITE_MT :
case DASD_ECKD_CCW_WRITE_KD :
case DASD_ECKD_CCW_WRITE_KD_MT :
2009-03-26 15:23:48 +01:00
dedata - > mask . perm = 0x02 ;
dedata - > attributes . operation = basepriv - > attrib . operation ;
2008-01-26 14:11:23 +01:00
rc = check_XRC_on_prefix ( pfxdata , basedev ) ;
break ;
case DASD_ECKD_CCW_WRITE_CKD :
case DASD_ECKD_CCW_WRITE_CKD_MT :
2009-03-26 15:23:48 +01:00
dedata - > attributes . operation = DASD_BYPASS_CACHE ;
2008-01-26 14:11:23 +01:00
rc = check_XRC_on_prefix ( pfxdata , basedev ) ;
break ;
case DASD_ECKD_CCW_ERASE :
case DASD_ECKD_CCW_WRITE_HOME_ADDRESS :
case DASD_ECKD_CCW_WRITE_RECORD_ZERO :
2009-03-26 15:23:48 +01:00
dedata - > mask . perm = 0x3 ;
dedata - > mask . auth = 0x1 ;
dedata - > attributes . operation = DASD_BYPASS_CACHE ;
2008-01-26 14:11:23 +01:00
rc = check_XRC_on_prefix ( pfxdata , basedev ) ;
break ;
2011-01-05 12:48:06 +01:00
case DASD_ECKD_CCW_WRITE_FULL_TRACK :
dedata - > mask . perm = 0x03 ;
dedata - > attributes . operation = basepriv - > attrib . operation ;
dedata - > blk_size = 0 ;
break ;
2009-03-26 15:23:48 +01:00
case DASD_ECKD_CCW_WRITE_TRACK_DATA :
dedata - > mask . perm = 0x02 ;
dedata - > attributes . operation = basepriv - > attrib . operation ;
dedata - > blk_size = blksize ;
rc = check_XRC_on_prefix ( pfxdata , basedev ) ;
2008-01-26 14:11:23 +01:00
break ;
2009-03-26 15:23:48 +01:00
default :
DBF_DEV_EVENT ( DBF_ERR , basedev ,
" PFX LRE unknown opcode 0x%x " , cmd ) ;
BUG ( ) ;
return - EINVAL ;
2008-01-26 14:11:23 +01:00
}
2009-03-26 15:23:48 +01:00
dedata - > attributes . mode = 0x3 ; /* ECKD */
2008-01-26 14:11:23 +01:00
if ( ( basepriv - > rdc_data . cu_type = = 0x2105 | |
basepriv - > rdc_data . cu_type = = 0x2107 | |
basepriv - > rdc_data . cu_type = = 0x1750 )
& & ! ( basepriv - > uses_cdl & & trk < 2 ) )
2009-03-26 15:23:48 +01:00
dedata - > ga_extended | = 0x40 ; /* Regular Data Format Mode */
2008-01-26 14:11:23 +01:00
2009-03-26 15:23:47 +01:00
heads = basepriv - > rdc_data . trk_per_cyl ;
begcyl = trk / heads ;
beghead = trk % heads ;
endcyl = totrk / heads ;
endhead = totrk % heads ;
2008-01-26 14:11:23 +01:00
/* check for sequential prestage - enhance cylinder range */
2009-03-26 15:23:48 +01:00
if ( dedata - > attributes . operation = = DASD_SEQ_PRESTAGE | |
dedata - > attributes . operation = = DASD_SEQ_ACCESS ) {
2008-01-26 14:11:23 +01:00
2009-03-26 15:23:47 +01:00
if ( endcyl + basepriv - > attrib . nr_cyl < basepriv - > real_cyl )
endcyl + = basepriv - > attrib . nr_cyl ;
2008-01-26 14:11:23 +01:00
else
2009-03-26 15:23:47 +01:00
endcyl = ( basepriv - > real_cyl - 1 ) ;
2008-01-26 14:11:23 +01:00
}
2009-03-26 15:23:48 +01:00
set_ch_t ( & dedata - > beg_ext , begcyl , beghead ) ;
set_ch_t ( & dedata - > end_ext , endcyl , endhead ) ;
if ( format = = 1 ) {
fill_LRE_data ( lredata , trk , rec_on_trk , count , cmd ,
basedev , blksize , tlf ) ;
}
2008-01-26 14:11:23 +01:00
return rc ;
}
2009-03-26 15:23:48 +01:00
static int prefix ( struct ccw1 * ccw , struct PFX_eckd_data * pfxdata ,
unsigned int trk , unsigned int totrk , int cmd ,
struct dasd_device * basedev , struct dasd_device * startdev )
{
return prefix_LRE ( ccw , pfxdata , trk , totrk , cmd , basedev , startdev ,
0 , 0 , 0 , 0 , 0 ) ;
}
2007-02-05 21:18:53 +01:00
static void
2009-03-26 15:23:47 +01:00
locate_record ( struct ccw1 * ccw , struct LO_eckd_data * data , unsigned int trk ,
unsigned int rec_on_trk , int no_rec , int cmd ,
2005-04-16 15:20:36 -07:00
struct dasd_device * device , int reclen )
{
struct dasd_eckd_private * private ;
int sector ;
int dn , d ;
2006-06-29 14:58:12 +02:00
2005-04-16 15:20:36 -07:00
private = ( struct dasd_eckd_private * ) device - > private ;
DBF_DEV_EVENT ( DBF_INFO , device ,
" Locate: trk %d, rec %d, no_rec %d, cmd %d, reclen %d " ,
trk , rec_on_trk , no_rec , cmd , reclen ) ;
ccw - > cmd_code = DASD_ECKD_CCW_LOCATE_RECORD ;
ccw - > flags = 0 ;
ccw - > count = 16 ;
ccw - > cda = ( __u32 ) __pa ( data ) ;
2008-01-26 14:11:23 +01:00
memset ( data , 0 , sizeof ( struct LO_eckd_data ) ) ;
2005-04-16 15:20:36 -07:00
sector = 0 ;
if ( rec_on_trk ) {
switch ( private - > rdc_data . dev_type ) {
case 0x3390 :
dn = ceil_quot ( reclen + 6 , 232 ) ;
d = 9 + ceil_quot ( reclen + 6 * ( dn + 1 ) , 34 ) ;
sector = ( 49 + ( rec_on_trk - 1 ) * ( 10 + d ) ) / 8 ;
break ;
case 0x3380 :
d = 7 + ceil_quot ( reclen + 12 , 32 ) ;
sector = ( 39 + ( rec_on_trk - 1 ) * ( 8 + d ) ) / 7 ;
break ;
}
}
data - > sector = sector ;
data - > count = no_rec ;
switch ( cmd ) {
case DASD_ECKD_CCW_WRITE_HOME_ADDRESS :
data - > operation . orientation = 0x3 ;
data - > operation . operation = 0x03 ;
break ;
case DASD_ECKD_CCW_READ_HOME_ADDRESS :
data - > operation . orientation = 0x3 ;
data - > operation . operation = 0x16 ;
break ;
case DASD_ECKD_CCW_WRITE_RECORD_ZERO :
data - > operation . orientation = 0x1 ;
data - > operation . operation = 0x03 ;
data - > count + + ;
break ;
case DASD_ECKD_CCW_READ_RECORD_ZERO :
data - > operation . orientation = 0x3 ;
data - > operation . operation = 0x16 ;
data - > count + + ;
break ;
case DASD_ECKD_CCW_WRITE :
case DASD_ECKD_CCW_WRITE_MT :
case DASD_ECKD_CCW_WRITE_KD :
case DASD_ECKD_CCW_WRITE_KD_MT :
data - > auxiliary . last_bytes_used = 0x1 ;
data - > length = reclen ;
data - > operation . operation = 0x01 ;
break ;
case DASD_ECKD_CCW_WRITE_CKD :
case DASD_ECKD_CCW_WRITE_CKD_MT :
data - > auxiliary . last_bytes_used = 0x1 ;
data - > length = reclen ;
data - > operation . operation = 0x03 ;
break ;
case DASD_ECKD_CCW_READ :
case DASD_ECKD_CCW_READ_MT :
case DASD_ECKD_CCW_READ_KD :
case DASD_ECKD_CCW_READ_KD_MT :
data - > auxiliary . last_bytes_used = 0x1 ;
data - > length = reclen ;
data - > operation . operation = 0x06 ;
break ;
case DASD_ECKD_CCW_READ_CKD :
case DASD_ECKD_CCW_READ_CKD_MT :
data - > auxiliary . last_bytes_used = 0x1 ;
data - > length = reclen ;
data - > operation . operation = 0x16 ;
break ;
case DASD_ECKD_CCW_READ_COUNT :
data - > operation . operation = 0x06 ;
break ;
case DASD_ECKD_CCW_ERASE :
data - > length = reclen ;
data - > auxiliary . last_bytes_used = 0x1 ;
data - > operation . operation = 0x0b ;
break ;
default :
2009-03-26 15:23:49 +01:00
DBF_DEV_EVENT ( DBF_ERR , device , " unknown locate record "
" opcode 0x%x " , cmd ) ;
2005-04-16 15:20:36 -07:00
}
2009-03-26 15:23:47 +01:00
set_ch_t ( & data - > seek_addr ,
trk / private - > rdc_data . trk_per_cyl ,
trk % private - > rdc_data . trk_per_cyl ) ;
data - > search_arg . cyl = data - > seek_addr . cyl ;
data - > search_arg . head = data - > seek_addr . head ;
2005-04-16 15:20:36 -07:00
data - > search_arg . record = rec_on_trk ;
}
/*
* Returns 1 if the block is one of the special blocks that needs
* to get read / written with the KD variant of the command .
* That is DASD_ECKD_READ_KD_MT instead of DASD_ECKD_READ_MT and
* DASD_ECKD_WRITE_KD_MT instead of DASD_ECKD_WRITE_MT .
* Luckily the KD variants differ only by one bit ( 0x08 ) from the
* normal variant . So don ' t wonder about code like :
* if ( dasd_eckd_cdl_special ( blk_per_trk , recid ) )
* ccw - > cmd_code | = 0x8 ;
*/
static inline int
dasd_eckd_cdl_special ( int blk_per_trk , int recid )
{
if ( recid < 3 )
return 1 ;
if ( recid < blk_per_trk )
return 0 ;
if ( recid < 2 * blk_per_trk )
return 1 ;
return 0 ;
}
/*
* Returns the record size for the special blocks of the cdl format .
* Only returns something useful if dasd_eckd_cdl_special is true
* for the recid .
*/
static inline int
dasd_eckd_cdl_reclen ( int recid )
{
if ( recid < 3 )
return sizes_trk0 [ recid ] ;
return LABEL_SIZE ;
}
2011-12-27 11:27:27 +01:00
/* create unique id from private structure. */
static void create_uid ( struct dasd_eckd_private * private )
2006-04-27 18:40:28 -07:00
{
2008-08-01 16:39:09 +02:00
int count ;
2011-12-27 11:27:27 +01:00
struct dasd_uid * uid ;
2006-04-27 18:40:28 -07:00
2010-05-17 10:00:11 +02:00
uid = & private - > uid ;
2006-04-27 18:40:28 -07:00
memset ( uid , 0 , sizeof ( struct dasd_uid ) ) ;
2008-08-01 16:39:09 +02:00
memcpy ( uid - > vendor , private - > ned - > HDA_manufacturer ,
2006-08-10 15:45:16 +02:00
sizeof ( uid - > vendor ) - 1 ) ;
2006-04-27 18:40:28 -07:00
EBCASC ( uid - > vendor , sizeof ( uid - > vendor ) - 1 ) ;
2008-08-01 16:39:09 +02:00
memcpy ( uid - > serial , private - > ned - > HDA_location ,
2006-08-10 15:45:16 +02:00
sizeof ( uid - > serial ) - 1 ) ;
2006-04-27 18:40:28 -07:00
EBCASC ( uid - > serial , sizeof ( uid - > serial ) - 1 ) ;
2008-08-01 16:39:09 +02:00
uid - > ssid = private - > gneq - > subsystemID ;
2009-08-18 11:18:35 -07:00
uid - > real_unit_addr = private - > ned - > unit_addr ;
2008-08-01 16:39:09 +02:00
if ( private - > sneq ) {
uid - > type = private - > sneq - > sua_flags ;
2008-01-26 14:11:23 +01:00
if ( uid - > type = = UA_BASE_PAV_ALIAS )
2008-08-01 16:39:09 +02:00
uid - > base_unit_addr = private - > sneq - > base_unit_addr ;
2008-01-26 14:11:23 +01:00
} else {
uid - > type = UA_BASE_DEVICE ;
}
2008-08-01 16:39:09 +02:00
if ( private - > vdsneq ) {
for ( count = 0 ; count < 16 ; count + + ) {
sprintf ( uid - > vduit + 2 * count , " %02x " ,
private - > vdsneq - > uit [ count ] ) ;
}
}
2011-12-27 11:27:27 +01:00
}
/*
* Generate device unique id that specifies the physical device .
*/
static int dasd_eckd_generate_uid ( struct dasd_device * device )
{
struct dasd_eckd_private * private ;
unsigned long flags ;
private = ( struct dasd_eckd_private * ) device - > private ;
if ( ! private )
return - ENODEV ;
if ( ! private - > ned | | ! private - > gneq )
return - ENODEV ;
spin_lock_irqsave ( get_ccwdev_lock ( device - > cdev ) , flags ) ;
create_uid ( private ) ;
2010-05-17 10:00:11 +02:00
spin_unlock_irqrestore ( get_ccwdev_lock ( device - > cdev ) , flags ) ;
2006-04-27 18:40:28 -07:00
return 0 ;
}
2010-05-17 10:00:11 +02:00
static int dasd_eckd_get_uid ( struct dasd_device * device , struct dasd_uid * uid )
{
struct dasd_eckd_private * private ;
unsigned long flags ;
if ( device - > private ) {
private = ( struct dasd_eckd_private * ) device - > private ;
spin_lock_irqsave ( get_ccwdev_lock ( device - > cdev ) , flags ) ;
* uid = private - > uid ;
spin_unlock_irqrestore ( get_ccwdev_lock ( device - > cdev ) , flags ) ;
return 0 ;
}
return - EINVAL ;
}
2011-12-27 11:27:27 +01:00
/*
* compare device UID with data of a given dasd_eckd_private structure
* return 0 for match
*/
static int dasd_eckd_compare_path_uid ( struct dasd_device * device ,
struct dasd_eckd_private * private )
{
struct dasd_uid device_uid ;
create_uid ( private ) ;
dasd_eckd_get_uid ( device , & device_uid ) ;
return memcmp ( & device_uid , & private - > uid , sizeof ( struct dasd_uid ) ) ;
}
2011-01-05 12:48:03 +01:00
static void dasd_eckd_fill_rcd_cqr ( struct dasd_device * device ,
struct dasd_ccw_req * cqr ,
__u8 * rcd_buffer ,
__u8 lpm )
2007-05-04 18:47:51 +02:00
{
struct ccw1 * ccw ;
2011-01-05 12:48:03 +01:00
/*
* buffer has to start with EBCDIC " V1.0 " to show
* support for virtual device SNEQ
*/
rcd_buffer [ 0 ] = 0xE5 ;
rcd_buffer [ 1 ] = 0xF1 ;
rcd_buffer [ 2 ] = 0x4B ;
rcd_buffer [ 3 ] = 0xF0 ;
2007-05-04 18:47:51 +02:00
ccw = cqr - > cpaddr ;
2011-01-05 12:48:03 +01:00
ccw - > cmd_code = DASD_ECKD_CCW_RCD ;
ccw - > flags = 0 ;
2007-05-04 18:47:51 +02:00
ccw - > cda = ( __u32 ) ( addr_t ) rcd_buffer ;
2011-01-05 12:48:03 +01:00
ccw - > count = DASD_ECKD_RCD_DATA_SIZE ;
cqr - > magic = DASD_ECKD_MAGIC ;
2007-05-04 18:47:51 +02:00
2008-01-26 14:11:23 +01:00
cqr - > startdev = device ;
cqr - > memdev = device ;
cqr - > block = NULL ;
2007-05-04 18:47:51 +02:00
cqr - > expires = 10 * HZ ;
cqr - > lpm = lpm ;
2009-12-07 12:51:51 +01:00
cqr - > retries = 256 ;
2013-01-30 09:49:40 +01:00
cqr - > buildclk = get_tod_clock ( ) ;
2007-05-04 18:47:51 +02:00
cqr - > status = DASD_CQR_FILLED ;
2011-01-05 12:48:03 +01:00
set_bit ( DASD_CQR_VERIFY_PATH , & cqr - > flags ) ;
}
2011-10-30 15:16:57 +01:00
/*
* Wakeup helper for read_conf
* if the cqr is not done and needs some error recovery
* the buffer has to be re - initialized with the EBCDIC " V1.0 "
* to show support for virtual device SNEQ
*/
static void read_conf_cb ( struct dasd_ccw_req * cqr , void * data )
{
struct ccw1 * ccw ;
__u8 * rcd_buffer ;
if ( cqr - > status ! = DASD_CQR_DONE ) {
ccw = cqr - > cpaddr ;
rcd_buffer = ( __u8 * ) ( ( addr_t ) ccw - > cda ) ;
memset ( rcd_buffer , 0 , sizeof ( * rcd_buffer ) ) ;
rcd_buffer [ 0 ] = 0xE5 ;
rcd_buffer [ 1 ] = 0xF1 ;
rcd_buffer [ 2 ] = 0x4B ;
rcd_buffer [ 3 ] = 0xF0 ;
}
dasd_wakeup_cb ( cqr , data ) ;
}
2011-01-05 12:48:03 +01:00
static int dasd_eckd_read_conf_immediately ( struct dasd_device * device ,
struct dasd_ccw_req * cqr ,
__u8 * rcd_buffer ,
__u8 lpm )
{
struct ciw * ciw ;
int rc ;
/*
* sanity check : scan for RCD command in extended SenseID data
* some devices do not support RCD
*/
ciw = ccw_device_get_ciw ( device - > cdev , CIW_TYPE_RCD ) ;
if ( ! ciw | | ciw - > cmd ! = DASD_ECKD_CCW_RCD )
return - EOPNOTSUPP ;
dasd_eckd_fill_rcd_cqr ( device , cqr , rcd_buffer , lpm ) ;
clear_bit ( DASD_CQR_FLAGS_USE_ERP , & cqr - > flags ) ;
2011-01-05 12:48:04 +01:00
set_bit ( DASD_CQR_ALLOW_SLOCK , & cqr - > flags ) ;
2011-01-05 12:48:03 +01:00
cqr - > retries = 5 ;
2011-10-30 15:16:57 +01:00
cqr - > callback = read_conf_cb ;
2011-01-05 12:48:03 +01:00
rc = dasd_sleep_on_immediatly ( cqr ) ;
return rc ;
2007-05-04 18:47:51 +02:00
}
static int dasd_eckd_read_conf_lpm ( struct dasd_device * device ,
void * * rcd_buffer ,
int * rcd_buffer_size , __u8 lpm )
{
struct ciw * ciw ;
char * rcd_buf = NULL ;
int ret ;
struct dasd_ccw_req * cqr ;
/*
2011-01-05 12:48:03 +01:00
* sanity check : scan for RCD command in extended SenseID data
* some devices do not support RCD
2007-05-04 18:47:51 +02:00
*/
ciw = ccw_device_get_ciw ( device - > cdev , CIW_TYPE_RCD ) ;
2011-01-05 12:48:03 +01:00
if ( ! ciw | | ciw - > cmd ! = DASD_ECKD_CCW_RCD ) {
2007-05-04 18:47:51 +02:00
ret = - EOPNOTSUPP ;
goto out_error ;
}
2011-01-05 12:48:03 +01:00
rcd_buf = kzalloc ( DASD_ECKD_RCD_DATA_SIZE , GFP_KERNEL | GFP_DMA ) ;
2007-05-04 18:47:51 +02:00
if ( ! rcd_buf ) {
ret = - ENOMEM ;
goto out_error ;
}
2011-01-05 12:48:03 +01:00
cqr = dasd_smalloc_request ( DASD_ECKD_MAGIC , 1 /* RCD */ ,
0 , /* use rcd_buf as data ara */
device ) ;
2007-05-04 18:47:51 +02:00
if ( IS_ERR ( cqr ) ) {
2011-01-05 12:48:03 +01:00
DBF_DEV_EVENT ( DBF_WARNING , device , " %s " ,
" Could not allocate RCD request " ) ;
ret = - ENOMEM ;
2007-05-04 18:47:51 +02:00
goto out_error ;
}
2011-01-05 12:48:03 +01:00
dasd_eckd_fill_rcd_cqr ( device , cqr , rcd_buf , lpm ) ;
2011-10-30 15:16:57 +01:00
cqr - > callback = read_conf_cb ;
2007-05-04 18:47:51 +02:00
ret = dasd_sleep_on ( cqr ) ;
/*
* on success we update the user input parms
*/
2008-01-26 14:11:23 +01:00
dasd_sfree_request ( cqr , cqr - > memdev ) ;
2007-05-04 18:47:51 +02:00
if ( ret )
goto out_error ;
2011-01-05 12:48:03 +01:00
* rcd_buffer_size = DASD_ECKD_RCD_DATA_SIZE ;
2007-05-04 18:47:51 +02:00
* rcd_buffer = rcd_buf ;
return 0 ;
out_error :
kfree ( rcd_buf ) ;
* rcd_buffer = NULL ;
* rcd_buffer_size = 0 ;
return ret ;
}
2008-08-01 16:39:09 +02:00
static int dasd_eckd_identify_conf_parts ( struct dasd_eckd_private * private )
{
struct dasd_sneq * sneq ;
int i , count ;
private - > ned = NULL ;
private - > sneq = NULL ;
private - > vdsneq = NULL ;
private - > gneq = NULL ;
count = private - > conf_len / sizeof ( struct dasd_sneq ) ;
sneq = ( struct dasd_sneq * ) private - > conf_data ;
for ( i = 0 ; i < count ; + + i ) {
if ( sneq - > flags . identifier = = 1 & & sneq - > format = = 1 )
private - > sneq = sneq ;
else if ( sneq - > flags . identifier = = 1 & & sneq - > format = = 4 )
private - > vdsneq = ( struct vd_sneq * ) sneq ;
else if ( sneq - > flags . identifier = = 2 )
private - > gneq = ( struct dasd_gneq * ) sneq ;
else if ( sneq - > flags . identifier = = 3 & & sneq - > res1 = = 1 )
private - > ned = ( struct dasd_ned * ) sneq ;
sneq + + ;
}
if ( ! private - > ned | | ! private - > gneq ) {
private - > ned = NULL ;
private - > sneq = NULL ;
private - > vdsneq = NULL ;
private - > gneq = NULL ;
return - EINVAL ;
}
return 0 ;
} ;
static unsigned char dasd_eckd_path_access ( void * conf_data , int conf_len )
{
struct dasd_gneq * gneq ;
int i , count , found ;
count = conf_len / sizeof ( * gneq ) ;
gneq = ( struct dasd_gneq * ) conf_data ;
found = 0 ;
for ( i = 0 ; i < count ; + + i ) {
if ( gneq - > flags . identifier = = 2 ) {
found = 1 ;
break ;
}
gneq + + ;
}
if ( found )
return ( ( char * ) gneq ) [ 18 ] & 0x07 ;
else
return 0 ;
}
static int dasd_eckd_read_conf ( struct dasd_device * device )
2005-04-16 15:20:36 -07:00
{
void * conf_data ;
int conf_len , conf_data_saved ;
2015-08-07 13:20:16 +02:00
int rc , path_err , pos ;
2011-01-05 12:48:03 +01:00
__u8 lpm , opm ;
2011-12-27 11:27:27 +01:00
struct dasd_eckd_private * private , path_private ;
2011-01-05 12:48:03 +01:00
struct dasd_path * path_data ;
2011-12-27 11:27:27 +01:00
struct dasd_uid * uid ;
char print_path_uid [ 60 ] , print_device_uid [ 60 ] ;
2005-04-16 15:20:36 -07:00
private = ( struct dasd_eckd_private * ) device - > private ;
2011-01-05 12:48:03 +01:00
path_data = & device - > path_data ;
opm = ccw_device_get_path_mask ( device - > cdev ) ;
2005-04-16 15:20:36 -07:00
conf_data_saved = 0 ;
2012-11-27 17:04:14 +01:00
path_err = 0 ;
2005-04-16 15:20:36 -07:00
/* get configuration data per operational path */
for ( lpm = 0x80 ; lpm ; lpm > > = 1 ) {
2011-12-27 11:27:27 +01:00
if ( ! ( lpm & opm ) )
continue ;
rc = dasd_eckd_read_conf_lpm ( device , & conf_data ,
& conf_len , lpm ) ;
if ( rc & & rc ! = - EOPNOTSUPP ) { /* -EOPNOTSUPP is ok */
DBF_EVENT_DEVID ( DBF_WARNING , device - > cdev ,
" Read configuration data returned "
" error %d " , rc ) ;
return rc ;
}
if ( conf_data = = NULL ) {
DBF_EVENT_DEVID ( DBF_WARNING , device - > cdev , " %s " ,
" No configuration data "
" retrieved " ) ;
/* no further analysis possible */
path_data - > opm | = lpm ;
continue ; /* no error */
}
2015-08-07 13:20:16 +02:00
/* translate path mask to position in mask */
pos = 8 - ffs ( lpm ) ;
kfree ( private - > path_conf_data [ pos ] ) ;
if ( ( __u8 * ) private - > path_conf_data [ pos ] = =
private - > conf_data ) {
private - > conf_data = NULL ;
private - > conf_len = 0 ;
conf_data_saved = 0 ;
}
private - > path_conf_data [ pos ] =
( struct dasd_conf_data * ) conf_data ;
2011-12-27 11:27:27 +01:00
/* save first valid configuration data */
if ( ! conf_data_saved ) {
kfree ( private - > conf_data ) ;
private - > conf_data = conf_data ;
private - > conf_len = conf_len ;
if ( dasd_eckd_identify_conf_parts ( private ) ) {
private - > conf_data = NULL ;
private - > conf_len = 0 ;
kfree ( conf_data ) ;
continue ;
2005-04-16 15:20:36 -07:00
}
2011-12-27 11:27:27 +01:00
/*
* build device UID that other path data
* can be compared to it
*/
dasd_eckd_generate_uid ( device ) ;
conf_data_saved + + ;
} else {
path_private . conf_data = conf_data ;
path_private . conf_len = DASD_ECKD_RCD_DATA_SIZE ;
if ( dasd_eckd_identify_conf_parts (
& path_private ) ) {
path_private . conf_data = NULL ;
path_private . conf_len = 0 ;
kfree ( conf_data ) ;
continue ;
2005-04-16 15:20:36 -07:00
}
2011-12-27 11:27:27 +01:00
if ( dasd_eckd_compare_path_uid (
device , & path_private ) ) {
uid = & path_private . uid ;
if ( strlen ( uid - > vduit ) > 0 )
snprintf ( print_path_uid ,
sizeof ( print_path_uid ) ,
" %s.%s.%04x.%02x.%s " ,
uid - > vendor , uid - > serial ,
uid - > ssid , uid - > real_unit_addr ,
uid - > vduit ) ;
else
snprintf ( print_path_uid ,
sizeof ( print_path_uid ) ,
" %s.%s.%04x.%02x " ,
uid - > vendor , uid - > serial ,
uid - > ssid ,
uid - > real_unit_addr ) ;
uid = & private - > uid ;
if ( strlen ( uid - > vduit ) > 0 )
snprintf ( print_device_uid ,
sizeof ( print_device_uid ) ,
" %s.%s.%04x.%02x.%s " ,
uid - > vendor , uid - > serial ,
uid - > ssid , uid - > real_unit_addr ,
uid - > vduit ) ;
else
snprintf ( print_device_uid ,
sizeof ( print_device_uid ) ,
" %s.%s.%04x.%02x " ,
uid - > vendor , uid - > serial ,
uid - > ssid ,
uid - > real_unit_addr ) ;
dev_err ( & device - > cdev - > dev ,
" Not all channel paths lead to "
" the same device, path %02X leads to "
" device %s instead of %s \n " , lpm ,
print_path_uid , print_device_uid ) ;
2012-11-27 17:04:14 +01:00
path_err = - EINVAL ;
2014-10-01 14:39:47 +02:00
path_data - > cablepm | = lpm ;
2012-11-27 17:04:14 +01:00
continue ;
2005-04-16 15:20:36 -07:00
}
2011-12-27 11:27:27 +01:00
path_private . conf_data = NULL ;
path_private . conf_len = 0 ;
}
switch ( dasd_eckd_path_access ( conf_data , conf_len ) ) {
case 0x02 :
path_data - > npm | = lpm ;
break ;
case 0x03 :
path_data - > ppm | = lpm ;
break ;
2005-04-16 15:20:36 -07:00
}
2011-12-27 11:27:27 +01:00
path_data - > opm | = lpm ;
2014-10-01 14:39:47 +02:00
/*
* if the path is used
* it should not be in one of the negative lists
*/
path_data - > cablepm & = ~ lpm ;
path_data - > hpfpm & = ~ lpm ;
path_data - > cuirpm & = ~ lpm ;
2005-04-16 15:20:36 -07:00
}
2011-12-27 11:27:27 +01:00
2012-11-27 17:04:14 +01:00
return path_err ;
2005-04-16 15:20:36 -07:00
}
2011-01-05 12:48:03 +01:00
static int verify_fcx_max_data ( struct dasd_device * device , __u8 lpm )
{
struct dasd_eckd_private * private ;
int mdc ;
u32 fcx_max_data ;
private = ( struct dasd_eckd_private * ) device - > private ;
if ( private - > fcx_max_data ) {
mdc = ccw_device_get_mdc ( device - > cdev , lpm ) ;
if ( ( mdc < 0 ) ) {
dev_warn ( & device - > cdev - > dev ,
" Detecting the maximum data size for zHPF "
" requests failed (rc=%d) for a new path %x \n " ,
mdc , lpm ) ;
return mdc ;
}
fcx_max_data = mdc * FCX_MAX_DATA_FACTOR ;
if ( fcx_max_data < private - > fcx_max_data ) {
dev_warn ( & device - > cdev - > dev ,
" The maximum data size for zHPF requests %u "
" on a new path %x is below the active maximum "
" %u \n " , fcx_max_data , lpm ,
private - > fcx_max_data ) ;
return - EACCES ;
}
}
return 0 ;
}
2011-12-27 11:27:27 +01:00
static int rebuild_device_uid ( struct dasd_device * device ,
struct path_verification_work_data * data )
{
struct dasd_eckd_private * private ;
struct dasd_path * path_data ;
__u8 lpm , opm ;
int rc ;
rc = - ENODEV ;
private = ( struct dasd_eckd_private * ) device - > private ;
path_data = & device - > path_data ;
opm = device - > path_data . opm ;
for ( lpm = 0x80 ; lpm ; lpm > > = 1 ) {
if ( ! ( lpm & opm ) )
continue ;
memset ( & data - > rcd_buffer , 0 , sizeof ( data - > rcd_buffer ) ) ;
memset ( & data - > cqr , 0 , sizeof ( data - > cqr ) ) ;
data - > cqr . cpaddr = & data - > ccw ;
rc = dasd_eckd_read_conf_immediately ( device , & data - > cqr ,
data - > rcd_buffer ,
lpm ) ;
if ( rc ) {
if ( rc = = - EOPNOTSUPP ) /* -EOPNOTSUPP is ok */
continue ;
DBF_EVENT_DEVID ( DBF_WARNING , device - > cdev ,
" Read configuration data "
" returned error %d " , rc ) ;
break ;
}
memcpy ( private - > conf_data , data - > rcd_buffer ,
DASD_ECKD_RCD_DATA_SIZE ) ;
if ( dasd_eckd_identify_conf_parts ( private ) ) {
rc = - ENODEV ;
} else /* first valid path is enough */
break ;
}
if ( ! rc )
rc = dasd_eckd_generate_uid ( device ) ;
return rc ;
}
2011-01-05 12:48:03 +01:00
static void do_path_verification_work ( struct work_struct * work )
{
struct path_verification_work_data * data ;
struct dasd_device * device ;
2011-12-27 11:27:27 +01:00
struct dasd_eckd_private path_private ;
struct dasd_uid * uid ;
__u8 path_rcd_buf [ DASD_ECKD_RCD_DATA_SIZE ] ;
2014-10-01 14:39:47 +02:00
__u8 lpm , opm , npm , ppm , epm , hpfpm , cablepm ;
2011-01-05 12:48:03 +01:00
unsigned long flags ;
2011-12-27 11:27:27 +01:00
char print_uid [ 60 ] ;
2011-01-05 12:48:03 +01:00
int rc ;
data = container_of ( work , struct path_verification_work_data , worker ) ;
device = data - > device ;
2011-10-30 15:17:09 +01:00
/* delay path verification until device was resumed */
if ( test_bit ( DASD_FLAG_SUSPENDED , & device - > flags ) ) {
schedule_work ( work ) ;
return ;
}
2015-08-07 13:19:03 +02:00
/* check if path verification already running and delay if so */
if ( test_and_set_bit ( DASD_FLAG_PATH_VERIFY , & device - > flags ) ) {
schedule_work ( work ) ;
return ;
}
2011-01-05 12:48:03 +01:00
opm = 0 ;
npm = 0 ;
ppm = 0 ;
epm = 0 ;
2014-10-01 14:39:47 +02:00
hpfpm = 0 ;
cablepm = 0 ;
2011-01-05 12:48:03 +01:00
for ( lpm = 0x80 ; lpm ; lpm > > = 1 ) {
2011-12-27 11:27:27 +01:00
if ( ! ( lpm & data - > tbvpm ) )
continue ;
memset ( & data - > rcd_buffer , 0 , sizeof ( data - > rcd_buffer ) ) ;
memset ( & data - > cqr , 0 , sizeof ( data - > cqr ) ) ;
data - > cqr . cpaddr = & data - > ccw ;
rc = dasd_eckd_read_conf_immediately ( device , & data - > cqr ,
data - > rcd_buffer ,
lpm ) ;
if ( ! rc ) {
switch ( dasd_eckd_path_access ( data - > rcd_buffer ,
DASD_ECKD_RCD_DATA_SIZE )
) {
case 0x02 :
npm | = lpm ;
break ;
case 0x03 :
ppm | = lpm ;
break ;
}
opm | = lpm ;
} else if ( rc = = - EOPNOTSUPP ) {
DBF_EVENT_DEVID ( DBF_WARNING , device - > cdev , " %s " ,
" path verification: No configuration "
" data retrieved " ) ;
opm | = lpm ;
} else if ( rc = = - EAGAIN ) {
DBF_EVENT_DEVID ( DBF_WARNING , device - > cdev , " %s " ,
2011-01-05 12:48:03 +01:00
" path verification: device is stopped, "
" try again later " ) ;
2011-12-27 11:27:27 +01:00
epm | = lpm ;
} else {
dev_warn ( & device - > cdev - > dev ,
" Reading device feature codes failed "
" (rc=%d) for new path %x \n " , rc , lpm ) ;
continue ;
}
if ( verify_fcx_max_data ( device , lpm ) ) {
opm & = ~ lpm ;
npm & = ~ lpm ;
ppm & = ~ lpm ;
2014-10-01 14:39:47 +02:00
hpfpm | = lpm ;
2011-12-27 11:27:27 +01:00
continue ;
}
/*
* save conf_data for comparison after
* rebuild_device_uid may have changed
* the original data
*/
memcpy ( & path_rcd_buf , data - > rcd_buffer ,
DASD_ECKD_RCD_DATA_SIZE ) ;
path_private . conf_data = ( void * ) & path_rcd_buf ;
path_private . conf_len = DASD_ECKD_RCD_DATA_SIZE ;
if ( dasd_eckd_identify_conf_parts ( & path_private ) ) {
path_private . conf_data = NULL ;
path_private . conf_len = 0 ;
continue ;
}
/*
* compare path UID with device UID only if at least
* one valid path is left
* in other case the device UID may have changed and
* the first working path UID will be used as device UID
*/
if ( device - > path_data . opm & &
dasd_eckd_compare_path_uid ( device , & path_private ) ) {
/*
* the comparison was not successful
* rebuild the device UID with at least one
* known path in case a z / VM hyperswap command
* has changed the device
*
* after this compare again
*
* if either the rebuild or the recompare fails
* the path can not be used
*/
if ( rebuild_device_uid ( device , data ) | |
dasd_eckd_compare_path_uid (
device , & path_private ) ) {
uid = & path_private . uid ;
if ( strlen ( uid - > vduit ) > 0 )
snprintf ( print_uid , sizeof ( print_uid ) ,
" %s.%s.%04x.%02x.%s " ,
uid - > vendor , uid - > serial ,
uid - > ssid , uid - > real_unit_addr ,
uid - > vduit ) ;
else
snprintf ( print_uid , sizeof ( print_uid ) ,
" %s.%s.%04x.%02x " ,
uid - > vendor , uid - > serial ,
uid - > ssid ,
uid - > real_unit_addr ) ;
dev_err ( & device - > cdev - > dev ,
" The newly added channel path %02X "
" will not be used because it leads "
" to a different device %s \n " ,
lpm , print_uid ) ;
2011-01-05 12:48:03 +01:00
opm & = ~ lpm ;
npm & = ~ lpm ;
ppm & = ~ lpm ;
2014-10-01 14:39:47 +02:00
cablepm | = lpm ;
2011-12-27 11:27:27 +01:00
continue ;
2011-01-05 12:48:03 +01:00
}
}
2011-12-27 11:27:27 +01:00
/*
* There is a small chance that a path is lost again between
* above path verification and the following modification of
* the device opm mask . We could avoid that race here by using
* yet another path mask , but we rather deal with this unlikely
* situation in dasd_start_IO .
*/
spin_lock_irqsave ( get_ccwdev_lock ( device - > cdev ) , flags ) ;
if ( ! device - > path_data . opm & & opm ) {
device - > path_data . opm = opm ;
2014-10-01 14:39:47 +02:00
device - > path_data . cablepm & = ~ opm ;
device - > path_data . cuirpm & = ~ opm ;
device - > path_data . hpfpm & = ~ opm ;
2011-12-27 11:27:27 +01:00
dasd_generic_path_operational ( device ) ;
2014-10-01 14:39:47 +02:00
} else {
2011-12-27 11:27:27 +01:00
device - > path_data . opm | = opm ;
2014-10-01 14:39:47 +02:00
device - > path_data . cablepm & = ~ opm ;
device - > path_data . cuirpm & = ~ opm ;
device - > path_data . hpfpm & = ~ opm ;
}
2011-12-27 11:27:27 +01:00
device - > path_data . npm | = npm ;
device - > path_data . ppm | = ppm ;
device - > path_data . tbvpm | = epm ;
2014-10-01 14:39:47 +02:00
device - > path_data . cablepm | = cablepm ;
device - > path_data . hpfpm | = hpfpm ;
2011-12-27 11:27:27 +01:00
spin_unlock_irqrestore ( get_ccwdev_lock ( device - > cdev ) , flags ) ;
2011-01-05 12:48:03 +01:00
}
2015-08-07 13:19:03 +02:00
clear_bit ( DASD_FLAG_PATH_VERIFY , & device - > flags ) ;
2011-01-05 12:48:03 +01:00
dasd_put_device ( device ) ;
if ( data - > isglobal )
mutex_unlock ( & dasd_path_verification_mutex ) ;
else
kfree ( data ) ;
}
static int dasd_eckd_verify_path ( struct dasd_device * device , __u8 lpm )
{
struct path_verification_work_data * data ;
data = kmalloc ( sizeof ( * data ) , GFP_ATOMIC | GFP_DMA ) ;
if ( ! data ) {
if ( mutex_trylock ( & dasd_path_verification_mutex ) ) {
data = path_verification_worker ;
data - > isglobal = 1 ;
} else
return - ENOMEM ;
} else {
memset ( data , 0 , sizeof ( * data ) ) ;
data - > isglobal = 0 ;
}
INIT_WORK ( & data - > worker , do_path_verification_work ) ;
dasd_get_device ( device ) ;
data - > device = device ;
data - > tbvpm = lpm ;
schedule_work ( & data - > worker ) ;
return 0 ;
}
2008-01-26 14:11:23 +01:00
static int dasd_eckd_read_features ( struct dasd_device * device )
{
struct dasd_psf_prssd_data * prssdp ;
struct dasd_rssd_features * features ;
struct dasd_ccw_req * cqr ;
struct ccw1 * ccw ;
int rc ;
struct dasd_eckd_private * private ;
private = ( struct dasd_eckd_private * ) device - > private ;
2009-09-22 22:58:52 +02:00
memset ( & private - > features , 0 , sizeof ( struct dasd_rssd_features ) ) ;
2009-09-11 10:28:29 +02:00
cqr = dasd_smalloc_request ( DASD_ECKD_MAGIC , 1 /* PSF */ + 1 /* RSSD */ ,
2008-01-26 14:11:23 +01:00
( sizeof ( struct dasd_psf_prssd_data ) +
sizeof ( struct dasd_rssd_features ) ) ,
device ) ;
if ( IS_ERR ( cqr ) ) {
2009-12-07 12:51:52 +01:00
DBF_EVENT_DEVID ( DBF_WARNING , device - > cdev , " %s " , " Could not "
" allocate initialization request " ) ;
2008-01-26 14:11:23 +01:00
return PTR_ERR ( cqr ) ;
}
cqr - > startdev = device ;
cqr - > memdev = device ;
cqr - > block = NULL ;
2009-12-07 12:51:51 +01:00
cqr - > retries = 256 ;
2008-01-26 14:11:23 +01:00
cqr - > expires = 10 * HZ ;
/* Prepare for Read Subsystem Data */
prssdp = ( struct dasd_psf_prssd_data * ) cqr - > data ;
memset ( prssdp , 0 , sizeof ( struct dasd_psf_prssd_data ) ) ;
prssdp - > order = PSF_ORDER_PRSSD ;
prssdp - > suborder = 0x41 ; /* Read Feature Codes */
/* all other bytes of prssdp must be zero */
ccw = cqr - > cpaddr ;
ccw - > cmd_code = DASD_ECKD_CCW_PSF ;
ccw - > count = sizeof ( struct dasd_psf_prssd_data ) ;
ccw - > flags | = CCW_FLAG_CC ;
ccw - > cda = ( __u32 ) ( addr_t ) prssdp ;
/* Read Subsystem Data - feature codes */
features = ( struct dasd_rssd_features * ) ( prssdp + 1 ) ;
memset ( features , 0 , sizeof ( struct dasd_rssd_features ) ) ;
ccw + + ;
ccw - > cmd_code = DASD_ECKD_CCW_RSSD ;
ccw - > count = sizeof ( struct dasd_rssd_features ) ;
ccw - > cda = ( __u32 ) ( addr_t ) features ;
2013-01-30 09:49:40 +01:00
cqr - > buildclk = get_tod_clock ( ) ;
2008-01-26 14:11:23 +01:00
cqr - > status = DASD_CQR_FILLED ;
rc = dasd_sleep_on ( cqr ) ;
if ( rc = = 0 ) {
prssdp = ( struct dasd_psf_prssd_data * ) cqr - > data ;
features = ( struct dasd_rssd_features * ) ( prssdp + 1 ) ;
memcpy ( & private - > features , features ,
sizeof ( struct dasd_rssd_features ) ) ;
2009-09-22 22:58:52 +02:00
} else
dev_warn ( & device - > cdev - > dev , " Reading device feature codes "
" failed with rc=%d \n " , rc ) ;
2008-01-26 14:11:23 +01:00
dasd_sfree_request ( cqr , cqr - > memdev ) ;
return rc ;
}
2006-06-29 15:08:18 +02:00
/*
* Build CP for Perform Subsystem Function - SSC .
*/
2009-03-26 15:23:48 +01:00
static struct dasd_ccw_req * dasd_eckd_build_psf_ssc ( struct dasd_device * device ,
int enable_pav )
2006-06-29 15:08:18 +02:00
{
2008-01-26 14:11:23 +01:00
struct dasd_ccw_req * cqr ;
struct dasd_psf_ssc_data * psf_ssc_data ;
struct ccw1 * ccw ;
2006-06-29 15:08:18 +02:00
2009-09-11 10:28:29 +02:00
cqr = dasd_smalloc_request ( DASD_ECKD_MAGIC , 1 /* PSF */ ,
2006-06-29 15:08:18 +02:00
sizeof ( struct dasd_psf_ssc_data ) ,
device ) ;
2008-01-26 14:11:23 +01:00
if ( IS_ERR ( cqr ) ) {
2009-03-26 15:23:49 +01:00
DBF_DEV_EVENT ( DBF_WARNING , device , " %s " ,
2006-06-29 15:08:18 +02:00
" Could not allocate PSF-SSC request " ) ;
2008-01-26 14:11:23 +01:00
return cqr ;
}
psf_ssc_data = ( struct dasd_psf_ssc_data * ) cqr - > data ;
psf_ssc_data - > order = PSF_ORDER_SSC ;
2009-12-07 12:51:50 +01:00
psf_ssc_data - > suborder = 0xc0 ;
2009-03-26 15:23:48 +01:00
if ( enable_pav ) {
2009-12-07 12:51:50 +01:00
psf_ssc_data - > suborder | = 0x08 ;
2009-03-26 15:23:48 +01:00
psf_ssc_data - > reserved [ 0 ] = 0x88 ;
}
2008-01-26 14:11:23 +01:00
ccw = cqr - > cpaddr ;
ccw - > cmd_code = DASD_ECKD_CCW_PSF ;
ccw - > cda = ( __u32 ) ( addr_t ) psf_ssc_data ;
ccw - > count = 66 ;
cqr - > startdev = device ;
cqr - > memdev = device ;
cqr - > block = NULL ;
2009-12-07 12:51:51 +01:00
cqr - > retries = 256 ;
2008-01-26 14:11:23 +01:00
cqr - > expires = 10 * HZ ;
2013-01-30 09:49:40 +01:00
cqr - > buildclk = get_tod_clock ( ) ;
2008-01-26 14:11:23 +01:00
cqr - > status = DASD_CQR_FILLED ;
return cqr ;
2006-06-29 15:08:18 +02:00
}
/*
* Perform Subsystem Function .
* It is necessary to trigger CIO for channel revalidation since this
* call might change behaviour of DASD devices .
*/
static int
2012-09-11 15:10:58 +02:00
dasd_eckd_psf_ssc ( struct dasd_device * device , int enable_pav ,
unsigned long flags )
2006-06-29 15:08:18 +02:00
{
2008-01-26 14:11:23 +01:00
struct dasd_ccw_req * cqr ;
int rc ;
2009-03-26 15:23:48 +01:00
cqr = dasd_eckd_build_psf_ssc ( device , enable_pav ) ;
2008-01-26 14:11:23 +01:00
if ( IS_ERR ( cqr ) )
return PTR_ERR ( cqr ) ;
2012-09-11 15:10:58 +02:00
/*
* set flags e . g . turn on failfast , to prevent blocking
* the calling function should handle failed requests
*/
cqr - > flags | = flags ;
2008-01-26 14:11:23 +01:00
rc = dasd_sleep_on ( cqr ) ;
if ( ! rc )
/* trigger CIO to reprobe devices */
css_schedule_reprobe ( ) ;
2012-09-11 15:10:58 +02:00
else if ( cqr - > intrc = = - EAGAIN )
rc = - EAGAIN ;
2008-01-26 14:11:23 +01:00
dasd_sfree_request ( cqr , cqr - > memdev ) ;
return rc ;
2006-06-29 15:08:18 +02:00
}
/*
* Valide storage server of current device .
*/
2012-09-11 15:10:58 +02:00
static int dasd_eckd_validate_server ( struct dasd_device * device ,
unsigned long flags )
2006-06-29 15:08:18 +02:00
{
int rc ;
2008-01-26 14:11:23 +01:00
struct dasd_eckd_private * private ;
2009-03-26 15:23:48 +01:00
int enable_pav ;
2006-06-29 15:08:18 +02:00
2012-01-18 18:03:40 +01:00
private = ( struct dasd_eckd_private * ) device - > private ;
if ( private - > uid . type = = UA_BASE_PAV_ALIAS | |
private - > uid . type = = UA_HYPER_PAV_ALIAS )
2012-09-11 15:10:58 +02:00
return 0 ;
2006-06-29 15:08:18 +02:00
if ( dasd_nopav | | MACHINE_IS_VM )
2009-03-26 15:23:48 +01:00
enable_pav = 0 ;
else
enable_pav = 1 ;
2012-09-11 15:10:58 +02:00
rc = dasd_eckd_psf_ssc ( device , enable_pav , flags ) ;
2009-12-07 12:51:51 +01:00
2006-08-24 13:22:36 +02:00
/* may be requested feature is not available on server,
* therefore just report error and go ahead */
2009-12-07 12:51:52 +01:00
DBF_EVENT_DEVID ( DBF_WARNING , device - > cdev , " PSF-SSC for SSID %04x "
" returned rc=%d " , private - > uid . ssid , rc ) ;
2012-09-11 15:10:58 +02:00
return rc ;
2006-06-29 15:08:18 +02:00
}
2012-01-18 18:03:41 +01:00
/*
* worker to do a validate server in case of a lost pathgroup
*/
static void dasd_eckd_do_validate_server ( struct work_struct * work )
{
struct dasd_device * device = container_of ( work , struct dasd_device ,
kick_validate ) ;
2013-02-19 09:30:05 +01:00
unsigned long flags = 0 ;
set_bit ( DASD_CQR_FLAGS_FAILFAST , & flags ) ;
if ( dasd_eckd_validate_server ( device , flags )
2012-09-11 15:10:58 +02:00
= = - EAGAIN ) {
/* schedule worker again if failed */
schedule_work ( & device - > kick_validate ) ;
return ;
}
2012-01-18 18:03:41 +01:00
dasd_put_device ( device ) ;
}
static void dasd_eckd_kick_validate_server ( struct dasd_device * device )
{
dasd_get_device ( device ) ;
2012-03-11 11:59:37 -04:00
/* exit if device not online or in offline processing */
if ( test_bit ( DASD_FLAG_OFFLINE , & device - > flags ) | |
device - > state < DASD_STATE_ONLINE ) {
dasd_put_device ( device ) ;
return ;
}
2012-01-18 18:03:41 +01:00
/* queue call to do_validate_server to the kernel event daemon. */
2015-04-02 12:27:25 +02:00
if ( ! schedule_work ( & device - > kick_validate ) )
dasd_put_device ( device ) ;
2012-01-18 18:03:41 +01:00
}
2011-01-05 12:48:02 +01:00
static u32 get_fcx_max_data ( struct dasd_device * device )
{
int tpm , mdc ;
int fcx_in_css , fcx_in_gneq , fcx_in_features ;
struct dasd_eckd_private * private ;
if ( dasd_nofcx )
return 0 ;
/* is transport mode supported? */
private = ( struct dasd_eckd_private * ) device - > private ;
fcx_in_css = css_general_characteristics . fcx ;
fcx_in_gneq = private - > gneq - > reserved2 [ 7 ] & 0x04 ;
fcx_in_features = private - > features . feature [ 40 ] & 0x80 ;
tpm = fcx_in_css & & fcx_in_gneq & & fcx_in_features ;
if ( ! tpm )
return 0 ;
mdc = ccw_device_get_mdc ( device - > cdev , 0 ) ;
if ( mdc < 0 ) {
dev_warn ( & device - > cdev - > dev , " Detecting the maximum supported "
" data size for zHPF requests failed \n " ) ;
return 0 ;
} else
return mdc * FCX_MAX_DATA_FACTOR ;
}
2006-04-27 18:40:28 -07:00
/*
* Check device characteristics .
* If the device is accessible using ECKD discipline , the device is enabled .
*/
2005-04-16 15:20:36 -07:00
static int
dasd_eckd_check_characteristics ( struct dasd_device * device )
{
struct dasd_eckd_private * private ;
2008-01-26 14:11:23 +01:00
struct dasd_block * block ;
2010-05-17 10:00:11 +02:00
struct dasd_uid temp_uid ;
2012-01-18 18:03:40 +01:00
int rc , i ;
2010-03-08 12:26:24 +01:00
int readonly ;
2010-08-09 18:13:00 +02:00
unsigned long value ;
2005-04-16 15:20:36 -07:00
2012-01-18 18:03:41 +01:00
/* setup work queue for validate server*/
INIT_WORK ( & device - > kick_validate , dasd_eckd_do_validate_server ) ;
2009-12-07 12:51:30 +01:00
if ( ! ccw_device_is_pathgroup ( device - > cdev ) ) {
dev_warn ( & device - > cdev - > dev ,
" A channel path group could not be established \n " ) ;
return - EIO ;
}
if ( ! ccw_device_is_multipath ( device - > cdev ) ) {
dev_info ( & device - > cdev - > dev ,
" The DASD is not operating in multipath mode \n " ) ;
}
2005-04-16 15:20:36 -07:00
private = ( struct dasd_eckd_private * ) device - > private ;
2009-06-12 10:26:37 +02:00
if ( ! private ) {
private = kzalloc ( sizeof ( * private ) , GFP_KERNEL | GFP_DMA ) ;
if ( ! private ) {
2009-03-26 15:23:49 +01:00
dev_warn ( & device - > cdev - > dev ,
" Allocating memory for private DASD data "
" failed \n " ) ;
2005-04-16 15:20:36 -07:00
return - ENOMEM ;
}
device - > private = ( void * ) private ;
2009-06-12 10:26:37 +02:00
} else {
memset ( private , 0 , sizeof ( * private ) ) ;
2005-04-16 15:20:36 -07:00
}
/* Invalidate status of initial analysis. */
private - > init_cqr_status = - 1 ;
/* Set default cache operations. */
private - > attrib . operation = DASD_NORMAL_CACHE ;
private - > attrib . nr_cyl = 0 ;
2006-06-29 15:08:18 +02:00
/* Read Configuration Data */
rc = dasd_eckd_read_conf ( device ) ;
if ( rc )
2008-01-26 14:11:23 +01:00
goto out_err1 ;
2006-06-29 15:08:18 +02:00
2010-08-09 18:13:00 +02:00
/* set default timeout */
device - > default_expires = DASD_EXPIRES ;
2013-01-30 09:26:12 +00:00
/* set default retry count */
device - > default_retries = DASD_RETRIES ;
2010-08-09 18:13:00 +02:00
if ( private - > gneq ) {
value = 1 ;
for ( i = 0 ; i < private - > gneq - > timeout . value ; i + + )
value = 10 * value ;
value = value * private - > gneq - > timeout . number ;
/* do not accept useless values */
if ( value ! = 0 & & value < = DASD_EXPIRES_MAX )
device - > default_expires = value ;
}
2010-05-17 10:00:11 +02:00
dasd_eckd_get_uid ( device , & temp_uid ) ;
if ( temp_uid . type = = UA_BASE_DEVICE ) {
2008-01-26 14:11:23 +01:00
block = dasd_alloc_block ( ) ;
if ( IS_ERR ( block ) ) {
2009-12-07 12:51:52 +01:00
DBF_EVENT_DEVID ( DBF_WARNING , device - > cdev , " %s " ,
" could not allocate dasd "
" block structure " ) ;
2008-01-26 14:11:23 +01:00
rc = PTR_ERR ( block ) ;
goto out_err1 ;
}
device - > block = block ;
block - > base = device ;
}
2012-01-18 18:03:40 +01:00
/* register lcu with alias handling, enable PAV */
rc = dasd_alias_make_device_known_to_lcu ( device ) ;
if ( rc )
2008-01-26 14:11:23 +01:00
goto out_err2 ;
2012-01-18 18:03:40 +01:00
2012-09-11 15:10:58 +02:00
dasd_eckd_validate_server ( device , 0 ) ;
2009-12-07 12:51:53 +01:00
/* device may report different configuration data after LCU setup */
rc = dasd_eckd_read_conf ( device ) ;
if ( rc )
goto out_err3 ;
2008-01-26 14:11:23 +01:00
/* Read Feature Codes */
2009-09-22 22:58:52 +02:00
dasd_eckd_read_features ( device ) ;
2006-06-29 15:08:18 +02:00
2005-04-16 15:20:36 -07:00
/* Read Device Characteristics */
2009-09-11 10:28:29 +02:00
rc = dasd_generic_read_dev_chars ( device , DASD_ECKD_MAGIC ,
& private - > rdc_data , 64 ) ;
2008-01-26 14:11:23 +01:00
if ( rc ) {
2009-12-07 12:51:52 +01:00
DBF_EVENT_DEVID ( DBF_WARNING , device - > cdev ,
" Read device characteristic failed, rc=%d " , rc ) ;
2008-01-26 14:11:23 +01:00
goto out_err3 ;
}
2011-08-03 16:44:23 +02:00
if ( ( device - > features & DASD_FEATURE_USERAW ) & &
! ( private - > rdc_data . facilities . RT_in_LR ) ) {
dev_err ( & device - > cdev - > dev , " The storage server does not "
" support raw-track access \n " ) ;
rc = - EINVAL ;
goto out_err3 ;
}
2010-09-20 11:44:00 +05:30
/* find the valid cylinder size */
2009-03-26 15:23:47 +01:00
if ( private - > rdc_data . no_cyl = = LV_COMPAT_CYL & &
private - > rdc_data . long_no_cyl )
private - > real_cyl = private - > rdc_data . long_no_cyl ;
else
private - > real_cyl = private - > rdc_data . no_cyl ;
2011-01-05 12:48:02 +01:00
private - > fcx_max_data = get_fcx_max_data ( device ) ;
2010-03-08 12:26:24 +01:00
readonly = dasd_device_is_ro ( device ) ;
if ( readonly )
set_bit ( DASD_FLAG_DEVICE_RO , & device - > flags ) ;
2009-03-26 15:23:49 +01:00
dev_info ( & device - > cdev - > dev , " New DASD %04X/%02X (CU %04X/%02X) "
2010-03-08 12:26:24 +01:00
" with %d cylinders, %d heads, %d sectors%s \n " ,
2009-03-26 15:23:49 +01:00
private - > rdc_data . dev_type ,
private - > rdc_data . dev_model ,
private - > rdc_data . cu_type ,
private - > rdc_data . cu_model . model ,
2009-06-12 10:26:37 +02:00
private - > real_cyl ,
2009-03-26 15:23:49 +01:00
private - > rdc_data . trk_per_cyl ,
2010-03-08 12:26:24 +01:00
private - > rdc_data . sec_per_trk ,
readonly ? " , read-only device " : " " ) ;
2008-01-26 14:11:23 +01:00
return 0 ;
out_err3 :
dasd_alias_disconnect_device_from_lcu ( device ) ;
out_err2 :
dasd_free_block ( device - > block ) ;
device - > block = NULL ;
out_err1 :
2008-08-01 16:39:09 +02:00
kfree ( private - > conf_data ) ;
2008-01-26 14:11:23 +01:00
kfree ( device - > private ) ;
device - > private = NULL ;
2006-04-27 18:40:28 -07:00
return rc ;
2005-04-16 15:20:36 -07:00
}
2008-01-26 14:11:23 +01:00
static void dasd_eckd_uncheck_device ( struct dasd_device * device )
{
2008-08-01 16:39:09 +02:00
struct dasd_eckd_private * private ;
2015-08-07 13:20:16 +02:00
int i ;
2008-08-01 16:39:09 +02:00
private = ( struct dasd_eckd_private * ) device - > private ;
2008-01-26 14:11:23 +01:00
dasd_alias_disconnect_device_from_lcu ( device ) ;
2008-08-01 16:39:09 +02:00
private - > ned = NULL ;
private - > sneq = NULL ;
private - > vdsneq = NULL ;
private - > gneq = NULL ;
private - > conf_len = 0 ;
2015-08-07 13:20:16 +02:00
for ( i = 0 ; i < 8 ; i + + ) {
kfree ( private - > path_conf_data [ i ] ) ;
if ( ( __u8 * ) private - > path_conf_data [ i ] = =
private - > conf_data ) {
private - > conf_data = NULL ;
private - > conf_len = 0 ;
}
private - > path_conf_data [ i ] = NULL ;
}
2008-08-01 16:39:09 +02:00
kfree ( private - > conf_data ) ;
private - > conf_data = NULL ;
2008-01-26 14:11:23 +01:00
}
2005-04-16 15:20:36 -07:00
static struct dasd_ccw_req *
dasd_eckd_analysis_ccw ( struct dasd_device * device )
{
struct dasd_eckd_private * private ;
struct eckd_count * count_data ;
struct LO_eckd_data * LO_data ;
struct dasd_ccw_req * cqr ;
struct ccw1 * ccw ;
int cplength , datasize ;
int i ;
private = ( struct dasd_eckd_private * ) device - > private ;
cplength = 8 ;
datasize = sizeof ( struct DE_eckd_data ) + 2 * sizeof ( struct LO_eckd_data ) ;
2009-09-11 10:28:29 +02:00
cqr = dasd_smalloc_request ( DASD_ECKD_MAGIC , cplength , datasize , device ) ;
2005-04-16 15:20:36 -07:00
if ( IS_ERR ( cqr ) )
return cqr ;
ccw = cqr - > cpaddr ;
/* Define extent for the first 3 tracks. */
define_extent ( ccw + + , cqr - > data , 0 , 2 ,
DASD_ECKD_CCW_READ_COUNT , device ) ;
2008-01-26 14:11:23 +01:00
LO_data = cqr - > data + sizeof ( struct DE_eckd_data ) ;
2005-04-16 15:20:36 -07:00
/* Locate record for the first 4 records on track 0. */
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
locate_record ( ccw + + , LO_data + + , 0 , 0 , 4 ,
DASD_ECKD_CCW_READ_COUNT , device , 0 ) ;
count_data = private - > count_area ;
for ( i = 0 ; i < 4 ; i + + ) {
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
ccw - > cmd_code = DASD_ECKD_CCW_READ_COUNT ;
ccw - > flags = 0 ;
ccw - > count = 8 ;
ccw - > cda = ( __u32 ) ( addr_t ) count_data ;
ccw + + ;
count_data + + ;
}
/* Locate record for the first record on track 2. */
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
locate_record ( ccw + + , LO_data + + , 2 , 0 , 1 ,
DASD_ECKD_CCW_READ_COUNT , device , 0 ) ;
/* Read count ccw. */
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
ccw - > cmd_code = DASD_ECKD_CCW_READ_COUNT ;
ccw - > flags = 0 ;
ccw - > count = 8 ;
ccw - > cda = ( __u32 ) ( addr_t ) count_data ;
2008-01-26 14:11:23 +01:00
cqr - > block = NULL ;
cqr - > startdev = device ;
cqr - > memdev = device ;
2009-12-07 12:51:51 +01:00
cqr - > retries = 255 ;
2013-01-30 09:49:40 +01:00
cqr - > buildclk = get_tod_clock ( ) ;
2005-04-16 15:20:36 -07:00
cqr - > status = DASD_CQR_FILLED ;
return cqr ;
}
2009-12-07 12:51:51 +01:00
/* differentiate between 'no record found' and any other error */
static int dasd_eckd_analysis_evaluation ( struct dasd_ccw_req * init_cqr )
{
char * sense ;
if ( init_cqr - > status = = DASD_CQR_DONE )
return INIT_CQR_OK ;
else if ( init_cqr - > status = = DASD_CQR_NEED_ERP | |
init_cqr - > status = = DASD_CQR_FAILED ) {
sense = dasd_get_sense ( & init_cqr - > irb ) ;
if ( sense & & ( sense [ 1 ] & SNS1_NO_REC_FOUND ) )
return INIT_CQR_UNFORMATTED ;
else
return INIT_CQR_ERROR ;
} else
return INIT_CQR_ERROR ;
}
2005-04-16 15:20:36 -07:00
/*
* This is the callback function for the init_analysis cqr . It saves
* the status of the initial analysis ccw before it frees it and kicks
* the device to continue the startup sequence . This will call
* dasd_eckd_do_analysis again ( if the devices has not been marked
* for deletion in the meantime ) .
*/
2009-12-07 12:51:51 +01:00
static void dasd_eckd_analysis_callback ( struct dasd_ccw_req * init_cqr ,
void * data )
2005-04-16 15:20:36 -07:00
{
struct dasd_eckd_private * private ;
struct dasd_device * device ;
2008-01-26 14:11:23 +01:00
device = init_cqr - > startdev ;
2005-04-16 15:20:36 -07:00
private = ( struct dasd_eckd_private * ) device - > private ;
2009-12-07 12:51:51 +01:00
private - > init_cqr_status = dasd_eckd_analysis_evaluation ( init_cqr ) ;
2005-04-16 15:20:36 -07:00
dasd_sfree_request ( init_cqr , device ) ;
dasd_kick_device ( device ) ;
}
2009-12-07 12:51:51 +01:00
static int dasd_eckd_start_analysis ( struct dasd_block * block )
2005-04-16 15:20:36 -07:00
{
struct dasd_ccw_req * init_cqr ;
2008-01-26 14:11:23 +01:00
init_cqr = dasd_eckd_analysis_ccw ( block - > base ) ;
2005-04-16 15:20:36 -07:00
if ( IS_ERR ( init_cqr ) )
return PTR_ERR ( init_cqr ) ;
init_cqr - > callback = dasd_eckd_analysis_callback ;
init_cqr - > callback_data = NULL ;
init_cqr - > expires = 5 * HZ ;
2009-12-07 12:51:51 +01:00
/* first try without ERP, so we can later handle unformatted
* devices as special case
*/
clear_bit ( DASD_CQR_FLAGS_USE_ERP , & init_cqr - > flags ) ;
init_cqr - > retries = 0 ;
2005-04-16 15:20:36 -07:00
dasd_add_request_head ( init_cqr ) ;
return - EAGAIN ;
}
2009-12-07 12:51:51 +01:00
static int dasd_eckd_end_analysis ( struct dasd_block * block )
2005-04-16 15:20:36 -07:00
{
2008-01-26 14:11:23 +01:00
struct dasd_device * device ;
2005-04-16 15:20:36 -07:00
struct dasd_eckd_private * private ;
struct eckd_count * count_area ;
unsigned int sb , blk_per_trk ;
int status , i ;
2009-12-07 12:51:51 +01:00
struct dasd_ccw_req * init_cqr ;
2005-04-16 15:20:36 -07:00
2008-01-26 14:11:23 +01:00
device = block - > base ;
2005-04-16 15:20:36 -07:00
private = ( struct dasd_eckd_private * ) device - > private ;
status = private - > init_cqr_status ;
private - > init_cqr_status = - 1 ;
2009-12-07 12:51:51 +01:00
if ( status = = INIT_CQR_ERROR ) {
/* try again, this time with full ERP */
init_cqr = dasd_eckd_analysis_ccw ( device ) ;
dasd_sleep_on ( init_cqr ) ;
status = dasd_eckd_analysis_evaluation ( init_cqr ) ;
dasd_sfree_request ( init_cqr , device ) ;
}
2011-01-05 12:48:06 +01:00
if ( device - > features & DASD_FEATURE_USERAW ) {
block - > bp_block = DASD_RAW_BLOCKSIZE ;
blk_per_trk = DASD_RAW_BLOCK_PER_TRACK ;
block - > s2b_shift = 3 ;
goto raw ;
}
2009-12-07 12:51:51 +01:00
if ( status = = INIT_CQR_UNFORMATTED ) {
dev_warn ( & device - > cdev - > dev , " The DASD is not formatted \n " ) ;
2005-04-16 15:20:36 -07:00
return - EMEDIUMTYPE ;
2009-12-07 12:51:51 +01:00
} else if ( status = = INIT_CQR_ERROR ) {
dev_err ( & device - > cdev - > dev ,
" Detecting the DASD disk layout failed because "
" of an I/O error \n " ) ;
return - EIO ;
2005-04-16 15:20:36 -07:00
}
private - > uses_cdl = 1 ;
/* Check Track 0 for Compatible Disk Layout */
count_area = NULL ;
for ( i = 0 ; i < 3 ; i + + ) {
if ( private - > count_area [ i ] . kl ! = 4 | |
2012-09-20 18:37:36 +02:00
private - > count_area [ i ] . dl ! = dasd_eckd_cdl_reclen ( i ) - 4 | |
private - > count_area [ i ] . cyl ! = 0 | |
private - > count_area [ i ] . head ! = count_area_head [ i ] | |
private - > count_area [ i ] . record ! = count_area_rec [ i ] ) {
2005-04-16 15:20:36 -07:00
private - > uses_cdl = 0 ;
break ;
}
}
if ( i = = 3 )
count_area = & private - > count_area [ 4 ] ;
if ( private - > uses_cdl = = 0 ) {
for ( i = 0 ; i < 5 ; i + + ) {
if ( ( private - > count_area [ i ] . kl ! = 0 ) | |
( private - > count_area [ i ] . dl ! =
2012-09-20 18:37:36 +02:00
private - > count_area [ 0 ] . dl ) | |
private - > count_area [ i ] . cyl ! = 0 | |
private - > count_area [ i ] . head ! = count_area_head [ i ] | |
private - > count_area [ i ] . record ! = count_area_rec [ i ] )
2005-04-16 15:20:36 -07:00
break ;
}
if ( i = = 5 )
count_area = & private - > count_area [ 0 ] ;
} else {
if ( private - > count_area [ 3 ] . record = = 1 )
2009-03-26 15:23:49 +01:00
dev_warn ( & device - > cdev - > dev ,
" Track 0 has no records following the VTOC \n " ) ;
2005-04-16 15:20:36 -07:00
}
2011-01-05 12:48:06 +01:00
2005-04-16 15:20:36 -07:00
if ( count_area ! = NULL & & count_area - > kl = = 0 ) {
/* we found notthing violating our disk layout */
if ( dasd_check_blocksize ( count_area - > dl ) = = 0 )
2008-01-26 14:11:23 +01:00
block - > bp_block = count_area - > dl ;
2005-04-16 15:20:36 -07:00
}
2008-01-26 14:11:23 +01:00
if ( block - > bp_block = = 0 ) {
2009-03-26 15:23:49 +01:00
dev_warn ( & device - > cdev - > dev ,
" The disk layout of the DASD is not supported \n " ) ;
2005-04-16 15:20:36 -07:00
return - EMEDIUMTYPE ;
}
2008-01-26 14:11:23 +01:00
block - > s2b_shift = 0 ; /* bits to shift 512 to get a block */
for ( sb = 512 ; sb < block - > bp_block ; sb = sb < < 1 )
block - > s2b_shift + + ;
2005-04-16 15:20:36 -07:00
2008-01-26 14:11:23 +01:00
blk_per_trk = recs_per_track ( & private - > rdc_data , 0 , block - > bp_block ) ;
2011-01-05 12:48:06 +01:00
raw :
2009-03-26 15:23:47 +01:00
block - > blocks = ( private - > real_cyl *
2005-04-16 15:20:36 -07:00
private - > rdc_data . trk_per_cyl *
blk_per_trk ) ;
2009-03-26 15:23:49 +01:00
dev_info ( & device - > cdev - > dev ,
" DASD with %d KB/block, %d KB total size, %d KB/track, "
" %s \n " , ( block - > bp_block > > 10 ) ,
( ( private - > real_cyl *
private - > rdc_data . trk_per_cyl *
blk_per_trk * ( block - > bp_block > > 9 ) ) > > 1 ) ,
( ( blk_per_trk * block - > bp_block ) > > 10 ) ,
private - > uses_cdl ?
" compatible disk layout " : " linux disk layout " ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2008-01-26 14:11:23 +01:00
static int dasd_eckd_do_analysis ( struct dasd_block * block )
2005-04-16 15:20:36 -07:00
{
struct dasd_eckd_private * private ;
2008-01-26 14:11:23 +01:00
private = ( struct dasd_eckd_private * ) block - > base - > private ;
2005-04-16 15:20:36 -07:00
if ( private - > init_cqr_status < 0 )
2008-01-26 14:11:23 +01:00
return dasd_eckd_start_analysis ( block ) ;
2005-04-16 15:20:36 -07:00
else
2008-01-26 14:11:23 +01:00
return dasd_eckd_end_analysis ( block ) ;
2005-04-16 15:20:36 -07:00
}
2013-04-15 16:22:23 +02:00
static int dasd_eckd_basic_to_ready ( struct dasd_device * device )
2008-01-26 14:11:23 +01:00
{
return dasd_alias_add_device ( device ) ;
} ;
static int dasd_eckd_online_to_ready ( struct dasd_device * device )
{
2010-05-17 10:00:10 +02:00
cancel_work_sync ( & device - > reload_device ) ;
2012-03-11 11:59:37 -04:00
cancel_work_sync ( & device - > kick_validate ) ;
2013-04-15 16:22:23 +02:00
return 0 ;
} ;
2014-07-18 14:19:25 +02:00
static int dasd_eckd_basic_to_known ( struct dasd_device * device )
2013-04-15 16:22:23 +02:00
{
2008-01-26 14:11:23 +01:00
return dasd_alias_remove_device ( device ) ;
} ;
2005-04-16 15:20:36 -07:00
static int
2008-01-26 14:11:23 +01:00
dasd_eckd_fill_geometry ( struct dasd_block * block , struct hd_geometry * geo )
2005-04-16 15:20:36 -07:00
{
struct dasd_eckd_private * private ;
2008-01-26 14:11:23 +01:00
private = ( struct dasd_eckd_private * ) block - > base - > private ;
if ( dasd_check_blocksize ( block - > bp_block ) = = 0 ) {
2005-04-16 15:20:36 -07:00
geo - > sectors = recs_per_track ( & private - > rdc_data ,
2008-01-26 14:11:23 +01:00
0 , block - > bp_block ) ;
2005-04-16 15:20:36 -07:00
}
geo - > cylinders = private - > rdc_data . no_cyl ;
geo - > heads = private - > rdc_data . trk_per_cyl ;
return 0 ;
}
static struct dasd_ccw_req *
2013-04-15 16:22:23 +02:00
dasd_eckd_build_format ( struct dasd_device * base ,
2014-07-18 14:26:01 +02:00
struct format_data_t * fdata ,
2014-07-28 10:16:35 +02:00
int enable_pav )
2005-04-16 15:20:36 -07:00
{
2013-04-15 16:22:23 +02:00
struct dasd_eckd_private * base_priv ;
struct dasd_eckd_private * start_priv ;
2014-07-18 14:26:01 +02:00
struct dasd_device * startdev = NULL ;
2005-04-16 15:20:36 -07:00
struct dasd_ccw_req * fcp ;
struct eckd_count * ect ;
2013-04-15 16:22:23 +02:00
struct ch_t address ;
2005-04-16 15:20:36 -07:00
struct ccw1 * ccw ;
void * data ;
2009-03-26 15:23:47 +01:00
int rpt ;
2005-04-16 15:20:36 -07:00
int cplength , datasize ;
2013-04-15 16:22:23 +02:00
int i , j ;
2009-03-26 15:23:46 +01:00
int intensity = 0 ;
int r0_perm ;
2013-04-15 16:22:23 +02:00
int nr_tracks ;
2013-10-09 14:30:08 +02:00
int use_prefix ;
2005-04-16 15:20:36 -07:00
2014-07-28 10:16:35 +02:00
if ( enable_pav )
2014-07-18 14:26:01 +02:00
startdev = dasd_alias_get_start_dev ( base ) ;
2013-04-15 16:22:23 +02:00
if ( ! startdev )
startdev = base ;
2005-04-16 15:20:36 -07:00
2013-04-15 16:22:23 +02:00
start_priv = ( struct dasd_eckd_private * ) startdev - > private ;
base_priv = ( struct dasd_eckd_private * ) base - > private ;
rpt = recs_per_track ( & base_priv - > rdc_data , 0 , fdata - > blksize ) ;
nr_tracks = fdata - > stop_unit - fdata - > start_unit + 1 ;
2005-04-16 15:20:36 -07:00
/*
* fdata - > intensity is a bit string that tells us what to do :
* Bit 0 : write record zero
* Bit 1 : write home address , currently not supported
* Bit 2 : invalidate tracks
* Bit 3 : use OS / 390 compatible disk layout ( cdl )
2009-03-26 15:23:46 +01:00
* Bit 4 : do not allow storage subsystem to modify record zero
2005-04-16 15:20:36 -07:00
* Only some bit combinations do make sense .
*/
2009-03-26 15:23:46 +01:00
if ( fdata - > intensity & 0x10 ) {
r0_perm = 0 ;
intensity = fdata - > intensity & ~ 0x10 ;
} else {
r0_perm = 1 ;
intensity = fdata - > intensity ;
}
2013-04-15 16:22:23 +02:00
2013-10-09 14:30:08 +02:00
use_prefix = base_priv - > features . feature [ 8 ] & 0x01 ;
2009-03-26 15:23:46 +01:00
switch ( intensity ) {
2005-04-16 15:20:36 -07:00
case 0x00 : /* Normal format */
case 0x08 : /* Normal format, use cdl. */
2013-04-15 16:22:23 +02:00
cplength = 2 + ( rpt * nr_tracks ) ;
2013-10-09 14:30:08 +02:00
if ( use_prefix )
datasize = sizeof ( struct PFX_eckd_data ) +
sizeof ( struct LO_eckd_data ) +
rpt * nr_tracks * sizeof ( struct eckd_count ) ;
else
datasize = sizeof ( struct DE_eckd_data ) +
sizeof ( struct LO_eckd_data ) +
rpt * nr_tracks * sizeof ( struct eckd_count ) ;
2005-04-16 15:20:36 -07:00
break ;
case 0x01 : /* Write record zero and format track. */
case 0x09 : /* Write record zero and format track, use cdl. */
2013-04-15 16:22:23 +02:00
cplength = 2 + rpt * nr_tracks ;
2013-10-09 14:30:08 +02:00
if ( use_prefix )
datasize = sizeof ( struct PFX_eckd_data ) +
sizeof ( struct LO_eckd_data ) +
sizeof ( struct eckd_count ) +
rpt * nr_tracks * sizeof ( struct eckd_count ) ;
else
datasize = sizeof ( struct DE_eckd_data ) +
sizeof ( struct LO_eckd_data ) +
sizeof ( struct eckd_count ) +
rpt * nr_tracks * sizeof ( struct eckd_count ) ;
2005-04-16 15:20:36 -07:00
break ;
case 0x04 : /* Invalidate track. */
case 0x0c : /* Invalidate track, use cdl. */
cplength = 3 ;
2013-10-09 14:30:08 +02:00
if ( use_prefix )
datasize = sizeof ( struct PFX_eckd_data ) +
sizeof ( struct LO_eckd_data ) +
sizeof ( struct eckd_count ) ;
else
datasize = sizeof ( struct DE_eckd_data ) +
sizeof ( struct LO_eckd_data ) +
sizeof ( struct eckd_count ) ;
2005-04-16 15:20:36 -07:00
break ;
default :
2013-04-15 16:22:23 +02:00
dev_warn ( & startdev - > cdev - > dev ,
" An I/O control call used incorrect flags 0x%x \n " ,
fdata - > intensity ) ;
2005-04-16 15:20:36 -07:00
return ERR_PTR ( - EINVAL ) ;
}
/* Allocate the format ccw request. */
2013-04-15 16:22:23 +02:00
fcp = dasd_smalloc_request ( DASD_ECKD_MAGIC , cplength ,
datasize , startdev ) ;
2005-04-16 15:20:36 -07:00
if ( IS_ERR ( fcp ) )
return fcp ;
2013-04-15 16:22:23 +02:00
start_priv - > count + + ;
2005-04-16 15:20:36 -07:00
data = fcp - > data ;
ccw = fcp - > cpaddr ;
2009-03-26 15:23:46 +01:00
switch ( intensity & ~ 0x08 ) {
2005-04-16 15:20:36 -07:00
case 0x00 : /* Normal format. */
2013-10-09 14:30:08 +02:00
if ( use_prefix ) {
prefix ( ccw + + , ( struct PFX_eckd_data * ) data ,
fdata - > start_unit , fdata - > stop_unit ,
DASD_ECKD_CCW_WRITE_CKD , base , startdev ) ;
/* grant subsystem permission to format R0 */
if ( r0_perm )
( ( struct PFX_eckd_data * ) data )
- > define_extent . ga_extended | = 0x04 ;
data + = sizeof ( struct PFX_eckd_data ) ;
} else {
define_extent ( ccw + + , ( struct DE_eckd_data * ) data ,
fdata - > start_unit , fdata - > stop_unit ,
DASD_ECKD_CCW_WRITE_CKD , startdev ) ;
/* grant subsystem permission to format R0 */
if ( r0_perm )
( ( struct DE_eckd_data * ) data )
- > ga_extended | = 0x04 ;
data + = sizeof ( struct DE_eckd_data ) ;
}
2005-04-16 15:20:36 -07:00
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
locate_record ( ccw + + , ( struct LO_eckd_data * ) data ,
2013-04-15 16:22:23 +02:00
fdata - > start_unit , 0 , rpt * nr_tracks ,
DASD_ECKD_CCW_WRITE_CKD , base ,
2005-04-16 15:20:36 -07:00
fdata - > blksize ) ;
data + = sizeof ( struct LO_eckd_data ) ;
break ;
case 0x01 : /* Write record zero + format track. */
2013-10-09 14:30:08 +02:00
if ( use_prefix ) {
prefix ( ccw + + , ( struct PFX_eckd_data * ) data ,
fdata - > start_unit , fdata - > stop_unit ,
DASD_ECKD_CCW_WRITE_RECORD_ZERO ,
base , startdev ) ;
data + = sizeof ( struct PFX_eckd_data ) ;
} else {
define_extent ( ccw + + , ( struct DE_eckd_data * ) data ,
fdata - > start_unit , fdata - > stop_unit ,
DASD_ECKD_CCW_WRITE_RECORD_ZERO , startdev ) ;
data + = sizeof ( struct DE_eckd_data ) ;
}
2005-04-16 15:20:36 -07:00
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
locate_record ( ccw + + , ( struct LO_eckd_data * ) data ,
2013-04-15 16:22:23 +02:00
fdata - > start_unit , 0 , rpt * nr_tracks + 1 ,
DASD_ECKD_CCW_WRITE_RECORD_ZERO , base ,
base - > block - > bp_block ) ;
2005-04-16 15:20:36 -07:00
data + = sizeof ( struct LO_eckd_data ) ;
break ;
case 0x04 : /* Invalidate track. */
2013-10-09 14:30:08 +02:00
if ( use_prefix ) {
prefix ( ccw + + , ( struct PFX_eckd_data * ) data ,
fdata - > start_unit , fdata - > stop_unit ,
DASD_ECKD_CCW_WRITE_CKD , base , startdev ) ;
data + = sizeof ( struct PFX_eckd_data ) ;
} else {
define_extent ( ccw + + , ( struct DE_eckd_data * ) data ,
fdata - > start_unit , fdata - > stop_unit ,
DASD_ECKD_CCW_WRITE_CKD , startdev ) ;
data + = sizeof ( struct DE_eckd_data ) ;
}
2005-04-16 15:20:36 -07:00
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
locate_record ( ccw + + , ( struct LO_eckd_data * ) data ,
fdata - > start_unit , 0 , 1 ,
2013-04-15 16:22:23 +02:00
DASD_ECKD_CCW_WRITE_CKD , base , 8 ) ;
2005-04-16 15:20:36 -07:00
data + = sizeof ( struct LO_eckd_data ) ;
break ;
}
2013-04-15 16:22:23 +02:00
for ( j = 0 ; j < nr_tracks ; j + + ) {
/* calculate cylinder and head for the current track */
set_ch_t ( & address ,
( fdata - > start_unit + j ) /
base_priv - > rdc_data . trk_per_cyl ,
( fdata - > start_unit + j ) %
base_priv - > rdc_data . trk_per_cyl ) ;
if ( intensity & 0x01 ) { /* write record zero */
2005-04-16 15:20:36 -07:00
ect = ( struct eckd_count * ) data ;
data + = sizeof ( struct eckd_count ) ;
2009-03-26 15:23:47 +01:00
ect - > cyl = address . cyl ;
ect - > head = address . head ;
2013-04-15 16:22:23 +02:00
ect - > record = 0 ;
2005-04-16 15:20:36 -07:00
ect - > kl = 0 ;
2013-04-15 16:22:23 +02:00
ect - > dl = 8 ;
2005-04-16 15:20:36 -07:00
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
2013-04-15 16:22:23 +02:00
ccw - > cmd_code = DASD_ECKD_CCW_WRITE_RECORD_ZERO ;
2005-04-16 15:20:36 -07:00
ccw - > flags = CCW_FLAG_SLI ;
ccw - > count = 8 ;
ccw - > cda = ( __u32 ) ( addr_t ) ect ;
ccw + + ;
}
2013-04-15 16:22:23 +02:00
if ( ( intensity & ~ 0x08 ) & 0x04 ) { /* erase track */
ect = ( struct eckd_count * ) data ;
data + = sizeof ( struct eckd_count ) ;
ect - > cyl = address . cyl ;
ect - > head = address . head ;
ect - > record = 1 ;
ect - > kl = 0 ;
ect - > dl = 0 ;
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
ccw - > cmd_code = DASD_ECKD_CCW_WRITE_CKD ;
ccw - > flags = CCW_FLAG_SLI ;
ccw - > count = 8 ;
ccw - > cda = ( __u32 ) ( addr_t ) ect ;
} else { /* write remaining records */
for ( i = 0 ; i < rpt ; i + + ) {
ect = ( struct eckd_count * ) data ;
data + = sizeof ( struct eckd_count ) ;
ect - > cyl = address . cyl ;
ect - > head = address . head ;
ect - > record = i + 1 ;
ect - > kl = 0 ;
ect - > dl = fdata - > blksize ;
/*
* Check for special tracks 0 - 1
* when formatting CDL
*/
if ( ( intensity & 0x08 ) & &
fdata - > start_unit = = 0 ) {
if ( i < 3 ) {
ect - > kl = 4 ;
ect - > dl = sizes_trk0 [ i ] - 4 ;
}
}
if ( ( intensity & 0x08 ) & &
fdata - > start_unit = = 1 ) {
ect - > kl = 44 ;
ect - > dl = LABEL_SIZE - 44 ;
}
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
if ( i ! = 0 | | j = = 0 )
ccw - > cmd_code =
DASD_ECKD_CCW_WRITE_CKD ;
else
ccw - > cmd_code =
DASD_ECKD_CCW_WRITE_CKD_MT ;
ccw - > flags = CCW_FLAG_SLI ;
ccw - > count = 8 ;
ccw - > cda = ( __u32 ) ( addr_t ) ect ;
ccw + + ;
}
}
2005-04-16 15:20:36 -07:00
}
2013-04-15 16:22:23 +02:00
fcp - > startdev = startdev ;
fcp - > memdev = startdev ;
2014-07-18 14:26:01 +02:00
fcp - > basedev = base ;
2009-12-07 12:51:51 +01:00
fcp - > retries = 256 ;
2013-04-15 16:22:23 +02:00
fcp - > expires = startdev - > default_expires * HZ ;
2013-01-30 09:49:40 +01:00
fcp - > buildclk = get_tod_clock ( ) ;
2005-04-16 15:20:36 -07:00
fcp - > status = DASD_CQR_FILLED ;
2013-04-15 16:22:23 +02:00
2005-04-16 15:20:36 -07:00
return fcp ;
}
2013-04-15 16:22:23 +02:00
static int
dasd_eckd_format_device ( struct dasd_device * base ,
2014-07-18 14:26:01 +02:00
struct format_data_t * fdata ,
2014-07-28 10:16:35 +02:00
int enable_pav )
2013-04-15 16:22:23 +02:00
{
struct dasd_ccw_req * cqr , * n ;
struct dasd_block * block ;
struct dasd_eckd_private * private ;
struct list_head format_queue ;
struct dasd_device * device ;
int old_stop , format_step ;
2014-07-18 14:26:01 +02:00
int step , rc = 0 , sleep_rc ;
2013-04-15 16:22:23 +02:00
block = base - > block ;
private = ( struct dasd_eckd_private * ) base - > private ;
/* Sanity checks. */
if ( fdata - > start_unit > =
( private - > real_cyl * private - > rdc_data . trk_per_cyl ) ) {
dev_warn ( & base - > cdev - > dev ,
" Start track number %u used in formatting is too big \n " ,
fdata - > start_unit ) ;
return - EINVAL ;
}
if ( fdata - > stop_unit > =
( private - > real_cyl * private - > rdc_data . trk_per_cyl ) ) {
dev_warn ( & base - > cdev - > dev ,
" Stop track number %u used in formatting is too big \n " ,
fdata - > stop_unit ) ;
return - EINVAL ;
}
if ( fdata - > start_unit > fdata - > stop_unit ) {
dev_warn ( & base - > cdev - > dev ,
" Start track %u used in formatting exceeds end track \n " ,
fdata - > start_unit ) ;
return - EINVAL ;
}
if ( dasd_check_blocksize ( fdata - > blksize ) ! = 0 ) {
dev_warn ( & base - > cdev - > dev ,
" The DASD cannot be formatted with block size %u \n " ,
fdata - > blksize ) ;
return - EINVAL ;
}
INIT_LIST_HEAD ( & format_queue ) ;
2014-07-18 14:26:01 +02:00
old_stop = fdata - > stop_unit ;
2013-04-15 16:22:23 +02:00
while ( fdata - > start_unit < = 1 ) {
fdata - > stop_unit = fdata - > start_unit ;
2014-07-28 10:16:35 +02:00
cqr = dasd_eckd_build_format ( base , fdata , enable_pav ) ;
2013-04-15 16:22:23 +02:00
list_add ( & cqr - > blocklist , & format_queue ) ;
fdata - > stop_unit = old_stop ;
fdata - > start_unit + + ;
if ( fdata - > start_unit > fdata - > stop_unit )
goto sleep ;
}
retry :
format_step = 255 / recs_per_track ( & private - > rdc_data , 0 ,
fdata - > blksize ) ;
while ( fdata - > start_unit < = old_stop ) {
step = fdata - > stop_unit - fdata - > start_unit + 1 ;
if ( step > format_step )
fdata - > stop_unit = fdata - > start_unit + format_step - 1 ;
2014-07-28 10:16:35 +02:00
cqr = dasd_eckd_build_format ( base , fdata , enable_pav ) ;
2013-04-15 16:22:23 +02:00
if ( IS_ERR ( cqr ) ) {
if ( PTR_ERR ( cqr ) = = - ENOMEM ) {
/*
* not enough memory available
* go to out and start requests
* retry after first requests were finished
*/
fdata - > stop_unit = old_stop ;
goto sleep ;
} else
return PTR_ERR ( cqr ) ;
}
list_add ( & cqr - > blocklist , & format_queue ) ;
fdata - > start_unit = fdata - > stop_unit + 1 ;
fdata - > stop_unit = old_stop ;
}
sleep :
2014-07-18 14:26:01 +02:00
sleep_rc = dasd_sleep_on_queue ( & format_queue ) ;
2013-04-15 16:22:23 +02:00
list_for_each_entry_safe ( cqr , n , & format_queue , blocklist ) {
device = cqr - > startdev ;
private = ( struct dasd_eckd_private * ) device - > private ;
if ( cqr - > status = = DASD_CQR_FAILED )
rc = - EIO ;
list_del_init ( & cqr - > blocklist ) ;
dasd_sfree_request ( cqr , device ) ;
private - > count - - ;
}
2014-07-18 14:26:01 +02:00
if ( sleep_rc )
return sleep_rc ;
2013-04-15 16:22:23 +02:00
/*
* in case of ENOMEM we need to retry after
* first requests are finished
*/
if ( fdata - > start_unit < = fdata - > stop_unit )
goto retry ;
return rc ;
}
2008-01-26 14:11:23 +01:00
static void dasd_eckd_handle_terminated_request ( struct dasd_ccw_req * cqr )
2005-04-16 15:20:36 -07:00
{
2013-01-30 09:26:14 +00:00
if ( cqr - > retries < 0 ) {
cqr - > status = DASD_CQR_FAILED ;
return ;
}
2008-01-26 14:11:23 +01:00
cqr - > status = DASD_CQR_FILLED ;
if ( cqr - > block & & ( cqr - > startdev ! = cqr - > block - > base ) ) {
dasd_eckd_reset_ccw_to_base_io ( cqr ) ;
cqr - > startdev = cqr - > block - > base ;
2011-01-05 12:48:03 +01:00
cqr - > lpm = cqr - > block - > base - > path_data . opm ;
2005-04-16 15:20:36 -07:00
}
2008-01-26 14:11:23 +01:00
} ;
2005-04-16 15:20:36 -07:00
static dasd_erp_fn_t
dasd_eckd_erp_action ( struct dasd_ccw_req * cqr )
{
2008-01-26 14:11:23 +01:00
struct dasd_device * device = ( struct dasd_device * ) cqr - > startdev ;
2005-04-16 15:20:36 -07:00
struct ccw_device * cdev = device - > cdev ;
switch ( cdev - > id . cu_type ) {
case 0x3990 :
case 0x2105 :
case 0x2107 :
case 0x1750 :
return dasd_3990_erp_action ;
case 0x9343 :
case 0x3880 :
default :
return dasd_default_erp_action ;
}
}
static dasd_erp_fn_t
dasd_eckd_erp_postaction ( struct dasd_ccw_req * cqr )
{
return dasd_default_erp_postaction ;
}
2011-01-05 12:48:04 +01:00
static void dasd_eckd_check_for_device_change ( struct dasd_device * device ,
struct dasd_ccw_req * cqr ,
struct irb * irb )
2008-01-26 14:11:23 +01:00
{
char mask ;
2009-03-26 15:23:48 +01:00
char * sense = NULL ;
2010-05-17 10:00:10 +02:00
struct dasd_eckd_private * private ;
2008-01-26 14:11:23 +01:00
2010-05-17 10:00:10 +02:00
private = ( struct dasd_eckd_private * ) device - > private ;
2008-01-26 14:11:23 +01:00
/* first of all check for state change pending interrupt */
mask = DEV_STAT_ATTENTION | DEV_STAT_DEV_END | DEV_STAT_UNIT_EXCEP ;
2009-03-26 15:23:48 +01:00
if ( ( scsw_dstat ( & irb - > scsw ) & mask ) = = mask ) {
2011-10-30 15:17:09 +01:00
/*
* for alias only , not in offline processing
* and only if not suspended
*/
2010-05-17 10:00:10 +02:00
if ( ! device - > block & & private - > lcu & &
2012-03-11 11:59:37 -04:00
device - > state = = DASD_STATE_ONLINE & &
2011-10-30 15:17:09 +01:00
! test_bit ( DASD_FLAG_OFFLINE , & device - > flags ) & &
! test_bit ( DASD_FLAG_SUSPENDED , & device - > flags ) ) {
2010-05-17 10:00:10 +02:00
/*
* the state change could be caused by an alias
* reassignment remove device from alias handling
* to prevent new requests from being scheduled on
* the wrong alias device
*/
dasd_alias_remove_device ( device ) ;
/* schedule worker to reload device */
dasd_reload_device ( device ) ;
}
2008-01-26 14:11:23 +01:00
dasd_generic_handle_state_change ( device ) ;
return ;
}
2010-10-25 16:10:47 +02:00
sense = dasd_get_sense ( irb ) ;
2011-01-05 12:48:04 +01:00
if ( ! sense )
return ;
/* summary unit check */
2011-04-20 10:15:35 +02:00
if ( ( sense [ 27 ] & DASD_SENSE_BIT_0 ) & & ( sense [ 7 ] = = 0x0D ) & &
2010-10-25 16:10:47 +02:00
( scsw_dstat ( & irb - > scsw ) & DEV_STAT_UNIT_CHECK ) ) {
2008-01-26 14:11:23 +01:00
dasd_alias_handle_summary_unit_check ( device , irb ) ;
return ;
}
2008-04-17 07:46:08 +02:00
/* service information message SIM */
2011-01-05 12:48:04 +01:00
if ( ! cqr & & ! ( sense [ 27 ] & DASD_SENSE_BIT_0 ) & &
2009-03-26 15:23:48 +01:00
( ( sense [ 6 ] & DASD_SIM_SENSE ) = = DASD_SIM_SENSE ) ) {
dasd_3990_erp_handle_sim ( device , sense ) ;
2008-04-17 07:46:08 +02:00
return ;
}
2011-01-05 12:48:04 +01:00
/* loss of device reservation is handled via base devices only
* as alias devices may be used with several bases
*/
2011-04-20 10:15:35 +02:00
if ( device - > block & & ( sense [ 27 ] & DASD_SENSE_BIT_0 ) & &
( sense [ 7 ] = = 0x3F ) & &
2011-01-05 12:48:04 +01:00
( scsw_dstat ( & irb - > scsw ) & DEV_STAT_UNIT_CHECK ) & &
test_bit ( DASD_FLAG_IS_RESERVED , & device - > flags ) ) {
if ( device - > features & DASD_FEATURE_FAILONSLCK )
set_bit ( DASD_FLAG_LOCK_STOLEN , & device - > flags ) ;
clear_bit ( DASD_FLAG_IS_RESERVED , & device - > flags ) ;
dev_err ( & device - > cdev - > dev ,
" The device reservation was lost \n " ) ;
2008-10-10 21:33:23 +02:00
}
2011-01-05 12:48:04 +01:00
}
2009-03-26 15:23:48 +01:00
static struct dasd_ccw_req * dasd_eckd_build_cp_cmd_single (
struct dasd_device * startdev ,
2008-01-26 14:11:23 +01:00
struct dasd_block * block ,
2009-03-26 15:23:48 +01:00
struct request * req ,
sector_t first_rec ,
sector_t last_rec ,
sector_t first_trk ,
sector_t last_trk ,
unsigned int first_offs ,
unsigned int last_offs ,
unsigned int blk_per_trk ,
unsigned int blksize )
2005-04-16 15:20:36 -07:00
{
struct dasd_eckd_private * private ;
unsigned long * idaws ;
struct LO_eckd_data * LO_data ;
struct dasd_ccw_req * cqr ;
struct ccw1 * ccw ;
2007-09-25 12:35:59 +02:00
struct req_iterator iter ;
2013-11-23 17:19:00 -08:00
struct bio_vec bv ;
2005-04-16 15:20:36 -07:00
char * dst ;
2009-03-26 15:23:48 +01:00
unsigned int off ;
2005-04-16 15:20:36 -07:00
int count , cidaw , cplength , datasize ;
2009-03-26 15:23:48 +01:00
sector_t recid ;
2005-04-16 15:20:36 -07:00
unsigned char cmd , rcmd ;
2008-01-26 14:11:23 +01:00
int use_prefix ;
struct dasd_device * basedev ;
2005-04-16 15:20:36 -07:00
2008-01-26 14:11:23 +01:00
basedev = block - > base ;
private = ( struct dasd_eckd_private * ) basedev - > private ;
2005-04-16 15:20:36 -07:00
if ( rq_data_dir ( req ) = = READ )
cmd = DASD_ECKD_CCW_READ_MT ;
else if ( rq_data_dir ( req ) = = WRITE )
cmd = DASD_ECKD_CCW_WRITE_MT ;
else
return ERR_PTR ( - EINVAL ) ;
2009-03-26 15:23:48 +01:00
2005-04-16 15:20:36 -07:00
/* Check struct bio and count the number of blocks for the request. */
count = 0 ;
cidaw = 0 ;
2007-09-25 12:35:59 +02:00
rq_for_each_segment ( bv , req , iter ) {
2013-11-23 17:19:00 -08:00
if ( bv . bv_len & ( blksize - 1 ) )
2007-08-16 13:43:12 +02:00
/* Eckd can only do full blocks. */
return ERR_PTR ( - EINVAL ) ;
2013-11-23 17:19:00 -08:00
count + = bv . bv_len > > ( block - > s2b_shift + 9 ) ;
if ( idal_is_needed ( page_address ( bv . bv_page ) , bv . bv_len ) )
cidaw + = bv . bv_len > > ( block - > s2b_shift + 9 ) ;
2005-04-16 15:20:36 -07:00
}
/* Paranoia. */
if ( count ! = last_rec - first_rec + 1 )
return ERR_PTR ( - EINVAL ) ;
2008-01-26 14:11:23 +01:00
/* use the prefix command if available */
use_prefix = private - > features . feature [ 8 ] & 0x01 ;
if ( use_prefix ) {
/* 1x prefix + number of blocks */
cplength = 2 + count ;
/* 1x prefix + cidaws*sizeof(long) */
datasize = sizeof ( struct PFX_eckd_data ) +
sizeof ( struct LO_eckd_data ) +
cidaw * sizeof ( unsigned long ) ;
} else {
/* 1x define extent + 1x locate record + number of blocks */
cplength = 2 + count ;
/* 1x define extent + 1x locate record + cidaws*sizeof(long) */
datasize = sizeof ( struct DE_eckd_data ) +
sizeof ( struct LO_eckd_data ) +
cidaw * sizeof ( unsigned long ) ;
}
2005-04-16 15:20:36 -07:00
/* Find out the number of additional locate record ccws for cdl. */
if ( private - > uses_cdl & & first_rec < 2 * blk_per_trk ) {
if ( last_rec > = 2 * blk_per_trk )
count = 2 * blk_per_trk - first_rec ;
cplength + = count ;
datasize + = count * sizeof ( struct LO_eckd_data ) ;
}
/* Allocate the ccw request. */
2009-09-11 10:28:29 +02:00
cqr = dasd_smalloc_request ( DASD_ECKD_MAGIC , cplength , datasize ,
startdev ) ;
2005-04-16 15:20:36 -07:00
if ( IS_ERR ( cqr ) )
return cqr ;
ccw = cqr - > cpaddr ;
2008-01-26 14:11:23 +01:00
/* First ccw is define extent or prefix. */
if ( use_prefix ) {
if ( prefix ( ccw + + , cqr - > data , first_trk ,
last_trk , cmd , basedev , startdev ) = = - EAGAIN ) {
/* Clock not in sync and XRC is enabled.
* Try again later .
*/
dasd_sfree_request ( cqr , startdev ) ;
return ERR_PTR ( - EAGAIN ) ;
}
idaws = ( unsigned long * ) ( cqr - > data +
sizeof ( struct PFX_eckd_data ) ) ;
} else {
if ( define_extent ( ccw + + , cqr - > data , first_trk ,
2011-12-27 11:27:28 +01:00
last_trk , cmd , basedev ) = = - EAGAIN ) {
2008-01-26 14:11:23 +01:00
/* Clock not in sync and XRC is enabled.
* Try again later .
*/
dasd_sfree_request ( cqr , startdev ) ;
return ERR_PTR ( - EAGAIN ) ;
}
idaws = ( unsigned long * ) ( cqr - > data +
sizeof ( struct DE_eckd_data ) ) ;
2007-02-05 21:18:19 +01:00
}
2005-04-16 15:20:36 -07:00
/* Build locate_record+read/write/ccws. */
LO_data = ( struct LO_eckd_data * ) ( idaws + cidaw ) ;
recid = first_rec ;
if ( private - > uses_cdl = = 0 | | recid > 2 * blk_per_trk ) {
/* Only standard blocks so there is just one locate record. */
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
locate_record ( ccw + + , LO_data + + , first_trk , first_offs + 1 ,
2008-01-26 14:11:23 +01:00
last_rec - recid + 1 , cmd , basedev , blksize ) ;
2005-04-16 15:20:36 -07:00
}
2007-09-25 12:35:59 +02:00
rq_for_each_segment ( bv , req , iter ) {
2013-11-23 17:19:00 -08:00
dst = page_address ( bv . bv_page ) + bv . bv_offset ;
2005-04-16 15:20:36 -07:00
if ( dasd_page_cache ) {
char * copy = kmem_cache_alloc ( dasd_page_cache ,
2006-12-06 20:33:19 -08:00
GFP_DMA | __GFP_NOWARN ) ;
2005-04-16 15:20:36 -07:00
if ( copy & & rq_data_dir ( req ) = = WRITE )
2013-11-23 17:19:00 -08:00
memcpy ( copy + bv . bv_offset , dst , bv . bv_len ) ;
2005-04-16 15:20:36 -07:00
if ( copy )
2013-11-23 17:19:00 -08:00
dst = copy + bv . bv_offset ;
2005-04-16 15:20:36 -07:00
}
2013-11-23 17:19:00 -08:00
for ( off = 0 ; off < bv . bv_len ; off + = blksize ) {
2005-04-16 15:20:36 -07:00
sector_t trkid = recid ;
unsigned int recoffs = sector_div ( trkid , blk_per_trk ) ;
rcmd = cmd ;
count = blksize ;
/* Locate record for cdl special block ? */
if ( private - > uses_cdl & & recid < 2 * blk_per_trk ) {
if ( dasd_eckd_cdl_special ( blk_per_trk , recid ) ) {
rcmd | = 0x8 ;
count = dasd_eckd_cdl_reclen ( recid ) ;
2005-05-01 08:58:59 -07:00
if ( count < blksize & &
rq_data_dir ( req ) = = READ )
2005-04-16 15:20:36 -07:00
memset ( dst + count , 0xe5 ,
blksize - count ) ;
}
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
locate_record ( ccw + + , LO_data + + ,
trkid , recoffs + 1 ,
2008-01-26 14:11:23 +01:00
1 , rcmd , basedev , count ) ;
2005-04-16 15:20:36 -07:00
}
/* Locate record for standard blocks ? */
if ( private - > uses_cdl & & recid = = 2 * blk_per_trk ) {
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
locate_record ( ccw + + , LO_data + + ,
trkid , recoffs + 1 ,
last_rec - recid + 1 ,
2008-01-26 14:11:23 +01:00
cmd , basedev , count ) ;
2005-04-16 15:20:36 -07:00
}
/* Read/write ccw. */
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
ccw - > cmd_code = rcmd ;
ccw - > count = count ;
if ( idal_is_needed ( dst , blksize ) ) {
ccw - > cda = ( __u32 ) ( addr_t ) idaws ;
ccw - > flags = CCW_FLAG_IDA ;
idaws = idal_create_words ( idaws , dst , blksize ) ;
} else {
ccw - > cda = ( __u32 ) ( addr_t ) dst ;
ccw - > flags = 0 ;
}
ccw + + ;
dst + = blksize ;
recid + + ;
}
}
2009-01-09 12:14:51 +01:00
if ( blk_noretry_request ( req ) | |
block - > base - > features & DASD_FEATURE_FAILFAST )
2006-01-06 00:19:15 -08:00
set_bit ( DASD_CQR_FLAGS_FAILFAST , & cqr - > flags ) ;
2008-01-26 14:11:23 +01:00
cqr - > startdev = startdev ;
cqr - > memdev = startdev ;
cqr - > block = block ;
2010-08-09 18:13:00 +02:00
cqr - > expires = startdev - > default_expires * HZ ; /* default 5 minutes */
2011-01-05 12:48:03 +01:00
cqr - > lpm = startdev - > path_data . ppm ;
2013-01-30 09:26:12 +00:00
cqr - > retries = startdev - > default_retries ;
2013-01-30 09:49:40 +01:00
cqr - > buildclk = get_tod_clock ( ) ;
2005-04-16 15:20:36 -07:00
cqr - > status = DASD_CQR_FILLED ;
return cqr ;
}
2009-03-26 15:23:48 +01:00
static struct dasd_ccw_req * dasd_eckd_build_cp_cmd_track (
struct dasd_device * startdev ,
struct dasd_block * block ,
struct request * req ,
sector_t first_rec ,
sector_t last_rec ,
sector_t first_trk ,
sector_t last_trk ,
unsigned int first_offs ,
unsigned int last_offs ,
unsigned int blk_per_trk ,
unsigned int blksize )
{
unsigned long * idaws ;
struct dasd_ccw_req * cqr ;
struct ccw1 * ccw ;
struct req_iterator iter ;
2013-11-23 17:19:00 -08:00
struct bio_vec bv ;
2009-03-26 15:23:48 +01:00
char * dst , * idaw_dst ;
unsigned int cidaw , cplength , datasize ;
unsigned int tlf ;
sector_t recid ;
unsigned char cmd ;
struct dasd_device * basedev ;
unsigned int trkcount , count , count_to_trk_end ;
unsigned int idaw_len , seg_len , part_len , len_to_track_end ;
unsigned char new_track , end_idaw ;
sector_t trkid ;
unsigned int recoffs ;
basedev = block - > base ;
if ( rq_data_dir ( req ) = = READ )
cmd = DASD_ECKD_CCW_READ_TRACK_DATA ;
else if ( rq_data_dir ( req ) = = WRITE )
cmd = DASD_ECKD_CCW_WRITE_TRACK_DATA ;
else
return ERR_PTR ( - EINVAL ) ;
/* Track based I/O needs IDAWs for each page, and not just for
* 64 bit addresses . We need additional idals for pages
* that get filled from two tracks , so we use the number
* of records as upper limit .
*/
cidaw = last_rec - first_rec + 1 ;
trkcount = last_trk - first_trk + 1 ;
/* 1x prefix + one read/write ccw per track */
cplength = 1 + trkcount ;
/* on 31-bit we need space for two 32 bit addresses per page
* on 64 - bit one 64 bit address
*/
datasize = sizeof ( struct PFX_eckd_data ) +
cidaw * sizeof ( unsigned long long ) ;
/* Allocate the ccw request. */
2009-09-11 10:28:29 +02:00
cqr = dasd_smalloc_request ( DASD_ECKD_MAGIC , cplength , datasize ,
startdev ) ;
2009-03-26 15:23:48 +01:00
if ( IS_ERR ( cqr ) )
return cqr ;
ccw = cqr - > cpaddr ;
/* transfer length factor: how many bytes to read from the last track */
if ( first_trk = = last_trk )
tlf = last_offs - first_offs + 1 ;
else
tlf = last_offs + 1 ;
tlf * = blksize ;
if ( prefix_LRE ( ccw + + , cqr - > data , first_trk ,
last_trk , cmd , basedev , startdev ,
1 /* format */ , first_offs + 1 ,
trkcount , blksize ,
tlf ) = = - EAGAIN ) {
/* Clock not in sync and XRC is enabled.
* Try again later .
*/
dasd_sfree_request ( cqr , startdev ) ;
return ERR_PTR ( - EAGAIN ) ;
}
/*
* The translation of request into ccw programs must meet the
* following conditions :
* - all idaws but the first and the last must address full pages
* ( or 2 K blocks on 31 - bit )
* - the scope of a ccw and it ' s idal ends with the track boundaries
*/
idaws = ( unsigned long * ) ( cqr - > data + sizeof ( struct PFX_eckd_data ) ) ;
recid = first_rec ;
new_track = 1 ;
end_idaw = 0 ;
len_to_track_end = 0 ;
2011-10-30 15:17:16 +01:00
idaw_dst = NULL ;
2009-03-26 15:23:48 +01:00
idaw_len = 0 ;
rq_for_each_segment ( bv , req , iter ) {
2013-11-23 17:19:00 -08:00
dst = page_address ( bv . bv_page ) + bv . bv_offset ;
seg_len = bv . bv_len ;
2009-03-26 15:23:48 +01:00
while ( seg_len ) {
if ( new_track ) {
trkid = recid ;
recoffs = sector_div ( trkid , blk_per_trk ) ;
count_to_trk_end = blk_per_trk - recoffs ;
count = min ( ( last_rec - recid + 1 ) ,
( sector_t ) count_to_trk_end ) ;
len_to_track_end = count * blksize ;
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
ccw - > cmd_code = cmd ;
ccw - > count = len_to_track_end ;
ccw - > cda = ( __u32 ) ( addr_t ) idaws ;
ccw - > flags = CCW_FLAG_IDA ;
ccw + + ;
recid + = count ;
new_track = 0 ;
2009-04-14 15:36:24 +02:00
/* first idaw for a ccw may start anywhere */
if ( ! idaw_dst )
idaw_dst = dst ;
2009-03-26 15:23:48 +01:00
}
2009-04-14 15:36:24 +02:00
/* If we start a new idaw, we must make sure that it
* starts on an IDA_BLOCK_SIZE boundary .
2009-03-26 15:23:48 +01:00
* If we continue an idaw , we must make sure that the
* current segment begins where the so far accumulated
* idaw ends
*/
2009-04-14 15:36:24 +02:00
if ( ! idaw_dst ) {
if ( __pa ( dst ) & ( IDA_BLOCK_SIZE - 1 ) ) {
dasd_sfree_request ( cqr , startdev ) ;
return ERR_PTR ( - ERANGE ) ;
} else
idaw_dst = dst ;
}
2009-03-26 15:23:48 +01:00
if ( ( idaw_dst + idaw_len ) ! = dst ) {
dasd_sfree_request ( cqr , startdev ) ;
return ERR_PTR ( - ERANGE ) ;
}
part_len = min ( seg_len , len_to_track_end ) ;
seg_len - = part_len ;
dst + = part_len ;
idaw_len + = part_len ;
len_to_track_end - = part_len ;
/* collected memory area ends on an IDA_BLOCK border,
* - > create an idaw
* idal_create_words will handle cases where idaw_len
* is larger then IDA_BLOCK_SIZE
*/
if ( ! ( __pa ( idaw_dst + idaw_len ) & ( IDA_BLOCK_SIZE - 1 ) ) )
end_idaw = 1 ;
/* We also need to end the idaw at track end */
if ( ! len_to_track_end ) {
new_track = 1 ;
end_idaw = 1 ;
}
if ( end_idaw ) {
idaws = idal_create_words ( idaws , idaw_dst ,
idaw_len ) ;
2011-10-30 15:17:16 +01:00
idaw_dst = NULL ;
2009-03-26 15:23:48 +01:00
idaw_len = 0 ;
end_idaw = 0 ;
}
}
}
if ( blk_noretry_request ( req ) | |
block - > base - > features & DASD_FEATURE_FAILFAST )
set_bit ( DASD_CQR_FLAGS_FAILFAST , & cqr - > flags ) ;
cqr - > startdev = startdev ;
cqr - > memdev = startdev ;
cqr - > block = block ;
2010-08-09 18:13:00 +02:00
cqr - > expires = startdev - > default_expires * HZ ; /* default 5 minutes */
2011-01-05 12:48:03 +01:00
cqr - > lpm = startdev - > path_data . ppm ;
2013-01-30 09:26:12 +00:00
cqr - > retries = startdev - > default_retries ;
2013-01-30 09:49:40 +01:00
cqr - > buildclk = get_tod_clock ( ) ;
2009-03-26 15:23:48 +01:00
cqr - > status = DASD_CQR_FILLED ;
return cqr ;
}
static int prepare_itcw ( struct itcw * itcw ,
unsigned int trk , unsigned int totrk , int cmd ,
struct dasd_device * basedev ,
struct dasd_device * startdev ,
unsigned int rec_on_trk , int count ,
unsigned int blksize ,
unsigned int total_data_size ,
unsigned int tlf ,
unsigned int blk_per_trk )
{
struct PFX_eckd_data pfxdata ;
struct dasd_eckd_private * basepriv , * startpriv ;
struct DE_eckd_data * dedata ;
struct LRE_eckd_data * lredata ;
struct dcw * dcw ;
u32 begcyl , endcyl ;
u16 heads , beghead , endhead ;
u8 pfx_cmd ;
int rc = 0 ;
int sector = 0 ;
int dn , d ;
/* setup prefix data */
basepriv = ( struct dasd_eckd_private * ) basedev - > private ;
startpriv = ( struct dasd_eckd_private * ) startdev - > private ;
dedata = & pfxdata . define_extent ;
lredata = & pfxdata . locate_record ;
memset ( & pfxdata , 0 , sizeof ( pfxdata ) ) ;
pfxdata . format = 1 ; /* PFX with LRE */
pfxdata . base_address = basepriv - > ned - > unit_addr ;
pfxdata . base_lss = basepriv - > ned - > ID ;
pfxdata . validity . define_extent = 1 ;
/* private uid is kept up to date, conf_data may be outdated */
if ( startpriv - > uid . type ! = UA_BASE_DEVICE ) {
pfxdata . validity . verify_base = 1 ;
if ( startpriv - > uid . type = = UA_HYPER_PAV_ALIAS )
pfxdata . validity . hyper_pav = 1 ;
}
switch ( cmd ) {
case DASD_ECKD_CCW_READ_TRACK_DATA :
dedata - > mask . perm = 0x1 ;
dedata - > attributes . operation = basepriv - > attrib . operation ;
dedata - > blk_size = blksize ;
dedata - > ga_extended | = 0x42 ;
lredata - > operation . orientation = 0x0 ;
lredata - > operation . operation = 0x0C ;
lredata - > auxiliary . check_bytes = 0x01 ;
pfx_cmd = DASD_ECKD_CCW_PFX_READ ;
break ;
case DASD_ECKD_CCW_WRITE_TRACK_DATA :
dedata - > mask . perm = 0x02 ;
dedata - > attributes . operation = basepriv - > attrib . operation ;
dedata - > blk_size = blksize ;
rc = check_XRC_on_prefix ( & pfxdata , basedev ) ;
dedata - > ga_extended | = 0x42 ;
lredata - > operation . orientation = 0x0 ;
lredata - > operation . operation = 0x3F ;
lredata - > extended_operation = 0x23 ;
lredata - > auxiliary . check_bytes = 0x2 ;
pfx_cmd = DASD_ECKD_CCW_PFX ;
break ;
default :
DBF_DEV_EVENT ( DBF_ERR , basedev ,
" prepare itcw, unknown opcode 0x%x " , cmd ) ;
BUG ( ) ;
break ;
}
if ( rc )
return rc ;
dedata - > attributes . mode = 0x3 ; /* ECKD */
heads = basepriv - > rdc_data . trk_per_cyl ;
begcyl = trk / heads ;
beghead = trk % heads ;
endcyl = totrk / heads ;
endhead = totrk % heads ;
/* check for sequential prestage - enhance cylinder range */
if ( dedata - > attributes . operation = = DASD_SEQ_PRESTAGE | |
dedata - > attributes . operation = = DASD_SEQ_ACCESS ) {
if ( endcyl + basepriv - > attrib . nr_cyl < basepriv - > real_cyl )
endcyl + = basepriv - > attrib . nr_cyl ;
else
endcyl = ( basepriv - > real_cyl - 1 ) ;
}
set_ch_t ( & dedata - > beg_ext , begcyl , beghead ) ;
set_ch_t ( & dedata - > end_ext , endcyl , endhead ) ;
dedata - > ep_format = 0x20 ; /* records per track is valid */
dedata - > ep_rec_per_track = blk_per_trk ;
if ( rec_on_trk ) {
switch ( basepriv - > rdc_data . dev_type ) {
case 0x3390 :
dn = ceil_quot ( blksize + 6 , 232 ) ;
d = 9 + ceil_quot ( blksize + 6 * ( dn + 1 ) , 34 ) ;
sector = ( 49 + ( rec_on_trk - 1 ) * ( 10 + d ) ) / 8 ;
break ;
case 0x3380 :
d = 7 + ceil_quot ( blksize + 12 , 32 ) ;
sector = ( 39 + ( rec_on_trk - 1 ) * ( 8 + d ) ) / 7 ;
break ;
}
}
lredata - > auxiliary . length_valid = 1 ;
lredata - > auxiliary . length_scope = 1 ;
lredata - > auxiliary . imbedded_ccw_valid = 1 ;
lredata - > length = tlf ;
lredata - > imbedded_ccw = cmd ;
lredata - > count = count ;
lredata - > sector = sector ;
set_ch_t ( & lredata - > seek_addr , begcyl , beghead ) ;
lredata - > search_arg . cyl = lredata - > seek_addr . cyl ;
lredata - > search_arg . head = lredata - > seek_addr . head ;
lredata - > search_arg . record = rec_on_trk ;
dcw = itcw_add_dcw ( itcw , pfx_cmd , 0 ,
& pfxdata , sizeof ( pfxdata ) , total_data_size ) ;
2013-06-01 11:52:16 +02:00
return PTR_RET ( dcw ) ;
2009-03-26 15:23:48 +01:00
}
static struct dasd_ccw_req * dasd_eckd_build_cp_tpm_track (
struct dasd_device * startdev ,
struct dasd_block * block ,
struct request * req ,
sector_t first_rec ,
sector_t last_rec ,
sector_t first_trk ,
sector_t last_trk ,
unsigned int first_offs ,
unsigned int last_offs ,
unsigned int blk_per_trk ,
unsigned int blksize )
{
struct dasd_ccw_req * cqr ;
struct req_iterator iter ;
2013-11-23 17:19:00 -08:00
struct bio_vec bv ;
2009-03-26 15:23:48 +01:00
char * dst ;
unsigned int trkcount , ctidaw ;
unsigned char cmd ;
struct dasd_device * basedev ;
unsigned int tlf ;
struct itcw * itcw ;
struct tidaw * last_tidaw = NULL ;
int itcw_op ;
size_t itcw_size ;
2011-01-05 12:48:02 +01:00
u8 tidaw_flags ;
unsigned int seg_len , part_len , len_to_track_end ;
unsigned char new_track ;
sector_t recid , trkid ;
unsigned int offs ;
unsigned int count , count_to_trk_end ;
2012-04-11 14:28:04 +02:00
int ret ;
2009-03-26 15:23:48 +01:00
basedev = block - > base ;
if ( rq_data_dir ( req ) = = READ ) {
cmd = DASD_ECKD_CCW_READ_TRACK_DATA ;
itcw_op = ITCW_OP_READ ;
} else if ( rq_data_dir ( req ) = = WRITE ) {
cmd = DASD_ECKD_CCW_WRITE_TRACK_DATA ;
itcw_op = ITCW_OP_WRITE ;
} else
return ERR_PTR ( - EINVAL ) ;
/* trackbased I/O needs address all memory via TIDAWs,
* not just for 64 bit addresses . This allows us to map
* each segment directly to one tidaw .
2011-01-05 12:48:02 +01:00
* In the case of write requests , additional tidaws may
* be needed when a segment crosses a track boundary .
2009-03-26 15:23:48 +01:00
*/
trkcount = last_trk - first_trk + 1 ;
ctidaw = 0 ;
rq_for_each_segment ( bv , req , iter ) {
+ + ctidaw ;
}
2011-01-05 12:48:02 +01:00
if ( rq_data_dir ( req ) = = WRITE )
ctidaw + = ( last_trk - first_trk ) ;
2009-03-26 15:23:48 +01:00
/* Allocate the ccw request. */
itcw_size = itcw_calc_size ( 0 , ctidaw , 0 ) ;
2009-09-11 10:28:29 +02:00
cqr = dasd_smalloc_request ( DASD_ECKD_MAGIC , 0 , itcw_size , startdev ) ;
2009-03-26 15:23:48 +01:00
if ( IS_ERR ( cqr ) )
return cqr ;
/* transfer length factor: how many bytes to read from the last track */
if ( first_trk = = last_trk )
tlf = last_offs - first_offs + 1 ;
else
tlf = last_offs + 1 ;
tlf * = blksize ;
itcw = itcw_init ( cqr - > data , itcw_size , itcw_op , 0 , ctidaw , 0 ) ;
2011-01-05 12:48:02 +01:00
if ( IS_ERR ( itcw ) ) {
2012-04-11 14:28:04 +02:00
ret = - EINVAL ;
goto out_error ;
2011-01-05 12:48:02 +01:00
}
2009-03-26 15:23:48 +01:00
cqr - > cpaddr = itcw_get_tcw ( itcw ) ;
if ( prepare_itcw ( itcw , first_trk , last_trk ,
cmd , basedev , startdev ,
first_offs + 1 ,
trkcount , blksize ,
( last_rec - first_rec + 1 ) * blksize ,
tlf , blk_per_trk ) = = - EAGAIN ) {
/* Clock not in sync and XRC is enabled.
* Try again later .
*/
2012-04-11 14:28:04 +02:00
ret = - EAGAIN ;
goto out_error ;
2009-03-26 15:23:48 +01:00
}
2011-03-15 17:08:21 +01:00
len_to_track_end = 0 ;
2009-03-26 15:23:48 +01:00
/*
* A tidaw can address 4 k of memory , but must not cross page boundaries
* We can let the block layer handle this by setting
* blk_queue_segment_boundary to page boundaries and
* blk_max_segment_size to page size when setting up the request queue .
2011-01-05 12:48:02 +01:00
* For write requests , a TIDAW must not cross track boundaries , because
* we have to set the CBC flag on the last tidaw for each track .
2009-03-26 15:23:48 +01:00
*/
2011-01-05 12:48:02 +01:00
if ( rq_data_dir ( req ) = = WRITE ) {
new_track = 1 ;
recid = first_rec ;
rq_for_each_segment ( bv , req , iter ) {
2013-11-23 17:19:00 -08:00
dst = page_address ( bv . bv_page ) + bv . bv_offset ;
seg_len = bv . bv_len ;
2011-01-05 12:48:02 +01:00
while ( seg_len ) {
if ( new_track ) {
trkid = recid ;
offs = sector_div ( trkid , blk_per_trk ) ;
count_to_trk_end = blk_per_trk - offs ;
count = min ( ( last_rec - recid + 1 ) ,
( sector_t ) count_to_trk_end ) ;
len_to_track_end = count * blksize ;
recid + = count ;
new_track = 0 ;
}
part_len = min ( seg_len , len_to_track_end ) ;
seg_len - = part_len ;
len_to_track_end - = part_len ;
/* We need to end the tidaw at track end */
if ( ! len_to_track_end ) {
new_track = 1 ;
tidaw_flags = TIDAW_FLAGS_INSERT_CBC ;
} else
tidaw_flags = 0 ;
last_tidaw = itcw_add_tidaw ( itcw , tidaw_flags ,
dst , part_len ) ;
2012-04-11 14:28:04 +02:00
if ( IS_ERR ( last_tidaw ) ) {
ret = - EINVAL ;
goto out_error ;
}
2011-01-05 12:48:02 +01:00
dst + = part_len ;
}
}
} else {
rq_for_each_segment ( bv , req , iter ) {
2013-11-23 17:19:00 -08:00
dst = page_address ( bv . bv_page ) + bv . bv_offset ;
2011-01-05 12:48:02 +01:00
last_tidaw = itcw_add_tidaw ( itcw , 0x00 ,
2013-11-23 17:19:00 -08:00
dst , bv . bv_len ) ;
2012-04-11 14:28:04 +02:00
if ( IS_ERR ( last_tidaw ) ) {
ret = - EINVAL ;
goto out_error ;
}
2011-01-05 12:48:02 +01:00
}
2009-03-26 15:23:48 +01:00
}
2011-01-05 12:48:02 +01:00
last_tidaw - > flags | = TIDAW_FLAGS_LAST ;
last_tidaw - > flags & = ~ TIDAW_FLAGS_INSERT_CBC ;
2009-03-26 15:23:48 +01:00
itcw_finalize ( itcw ) ;
if ( blk_noretry_request ( req ) | |
block - > base - > features & DASD_FEATURE_FAILFAST )
set_bit ( DASD_CQR_FLAGS_FAILFAST , & cqr - > flags ) ;
2011-01-05 12:48:02 +01:00
cqr - > cpmode = 1 ;
2009-03-26 15:23:48 +01:00
cqr - > startdev = startdev ;
cqr - > memdev = startdev ;
cqr - > block = block ;
2010-08-09 18:13:00 +02:00
cqr - > expires = startdev - > default_expires * HZ ; /* default 5 minutes */
2011-01-05 12:48:03 +01:00
cqr - > lpm = startdev - > path_data . ppm ;
2013-01-30 09:26:12 +00:00
cqr - > retries = startdev - > default_retries ;
2013-01-30 09:49:40 +01:00
cqr - > buildclk = get_tod_clock ( ) ;
2009-03-26 15:23:48 +01:00
cqr - > status = DASD_CQR_FILLED ;
return cqr ;
2012-04-11 14:28:04 +02:00
out_error :
dasd_sfree_request ( cqr , startdev ) ;
return ERR_PTR ( ret ) ;
2009-03-26 15:23:48 +01:00
}
static struct dasd_ccw_req * dasd_eckd_build_cp ( struct dasd_device * startdev ,
struct dasd_block * block ,
struct request * req )
{
2011-01-05 12:48:02 +01:00
int cmdrtd , cmdwtd ;
2009-03-26 15:23:48 +01:00
int use_prefix ;
2011-01-05 12:48:02 +01:00
int fcx_multitrack ;
2009-06-12 10:26:36 +02:00
struct dasd_eckd_private * private ;
2009-03-26 15:23:48 +01:00
struct dasd_device * basedev ;
sector_t first_rec , last_rec ;
sector_t first_trk , last_trk ;
unsigned int first_offs , last_offs ;
unsigned int blk_per_trk , blksize ;
int cdlspecial ;
2011-01-05 12:48:02 +01:00
unsigned int data_size ;
2009-03-26 15:23:48 +01:00
struct dasd_ccw_req * cqr ;
basedev = block - > base ;
private = ( struct dasd_eckd_private * ) basedev - > private ;
/* Calculate number of blocks/records per track. */
blksize = block - > bp_block ;
blk_per_trk = recs_per_track ( & private - > rdc_data , 0 , blksize ) ;
2009-10-06 10:34:15 +02:00
if ( blk_per_trk = = 0 )
return ERR_PTR ( - EINVAL ) ;
2009-03-26 15:23:48 +01:00
/* Calculate record id of first and last block. */
2009-05-07 22:24:39 +09:00
first_rec = first_trk = blk_rq_pos ( req ) > > block - > s2b_shift ;
2009-03-26 15:23:48 +01:00
first_offs = sector_div ( first_trk , blk_per_trk ) ;
last_rec = last_trk =
2009-05-07 22:24:39 +09:00
( blk_rq_pos ( req ) + blk_rq_sectors ( req ) - 1 ) > > block - > s2b_shift ;
2009-03-26 15:23:48 +01:00
last_offs = sector_div ( last_trk , blk_per_trk ) ;
cdlspecial = ( private - > uses_cdl & & first_rec < 2 * blk_per_trk ) ;
2011-01-05 12:48:02 +01:00
fcx_multitrack = private - > features . feature [ 40 ] & 0x20 ;
data_size = blk_rq_bytes ( req ) ;
2013-11-19 14:15:57 +01:00
if ( data_size % blksize )
return ERR_PTR ( - EINVAL ) ;
2011-01-05 12:48:02 +01:00
/* tpm write request add CBC data on each track boundary */
if ( rq_data_dir ( req ) = = WRITE )
data_size + = ( last_trk - first_trk ) * 4 ;
2009-03-26 15:23:48 +01:00
/* is read track data and write track data in command mode supported? */
cmdrtd = private - > features . feature [ 9 ] & 0x20 ;
cmdwtd = private - > features . feature [ 12 ] & 0x40 ;
use_prefix = private - > features . feature [ 8 ] & 0x01 ;
cqr = NULL ;
if ( cdlspecial | | dasd_page_cache ) {
/* do nothing, just fall through to the cmd mode single case */
2011-01-05 12:48:02 +01:00
} else if ( ( data_size < = private - > fcx_max_data )
& & ( fcx_multitrack | | ( first_trk = = last_trk ) ) ) {
2009-03-26 15:23:48 +01:00
cqr = dasd_eckd_build_cp_tpm_track ( startdev , block , req ,
first_rec , last_rec ,
first_trk , last_trk ,
first_offs , last_offs ,
blk_per_trk , blksize ) ;
2011-01-05 12:48:02 +01:00
if ( IS_ERR ( cqr ) & & ( PTR_ERR ( cqr ) ! = - EAGAIN ) & &
( PTR_ERR ( cqr ) ! = - ENOMEM ) )
2009-03-26 15:23:48 +01:00
cqr = NULL ;
} else if ( use_prefix & &
( ( ( rq_data_dir ( req ) = = READ ) & & cmdrtd ) | |
( ( rq_data_dir ( req ) = = WRITE ) & & cmdwtd ) ) ) {
cqr = dasd_eckd_build_cp_cmd_track ( startdev , block , req ,
first_rec , last_rec ,
first_trk , last_trk ,
first_offs , last_offs ,
blk_per_trk , blksize ) ;
2011-01-05 12:48:02 +01:00
if ( IS_ERR ( cqr ) & & ( PTR_ERR ( cqr ) ! = - EAGAIN ) & &
( PTR_ERR ( cqr ) ! = - ENOMEM ) )
2009-03-26 15:23:48 +01:00
cqr = NULL ;
}
if ( ! cqr )
cqr = dasd_eckd_build_cp_cmd_single ( startdev , block , req ,
first_rec , last_rec ,
first_trk , last_trk ,
first_offs , last_offs ,
blk_per_trk , blksize ) ;
return cqr ;
}
2011-01-05 12:48:06 +01:00
static struct dasd_ccw_req * dasd_raw_build_cp ( struct dasd_device * startdev ,
struct dasd_block * block ,
struct request * req )
{
unsigned long * idaws ;
struct dasd_device * basedev ;
struct dasd_ccw_req * cqr ;
struct ccw1 * ccw ;
struct req_iterator iter ;
2013-11-23 17:19:00 -08:00
struct bio_vec bv ;
2011-01-05 12:48:06 +01:00
char * dst ;
unsigned char cmd ;
unsigned int trkcount ;
unsigned int seg_len , len_to_track_end ;
unsigned int first_offs ;
unsigned int cidaw , cplength , datasize ;
2013-08-16 15:57:32 +02:00
sector_t first_trk , last_trk , sectors ;
sector_t start_padding_sectors , end_sector_offset , end_padding_sectors ;
2011-01-05 12:48:06 +01:00
unsigned int pfx_datasize ;
/*
* raw track access needs to be mutiple of 64 k and on 64 k boundary
2013-08-16 15:57:32 +02:00
* For read requests we can fix an incorrect alignment by padding
* the request with dummy pages .
2011-01-05 12:48:06 +01:00
*/
2013-08-16 15:57:32 +02:00
start_padding_sectors = blk_rq_pos ( req ) % DASD_RAW_SECTORS_PER_TRACK ;
end_sector_offset = ( blk_rq_pos ( req ) + blk_rq_sectors ( req ) ) %
DASD_RAW_SECTORS_PER_TRACK ;
end_padding_sectors = ( DASD_RAW_SECTORS_PER_TRACK - end_sector_offset ) %
DASD_RAW_SECTORS_PER_TRACK ;
basedev = block - > base ;
if ( ( start_padding_sectors | | end_padding_sectors ) & &
( rq_data_dir ( req ) = = WRITE ) ) {
DBF_DEV_EVENT ( DBF_ERR , basedev ,
" raw write not track aligned (%lu,%lu) req %p " ,
start_padding_sectors , end_padding_sectors , req ) ;
2011-01-05 12:48:06 +01:00
cqr = ERR_PTR ( - EINVAL ) ;
goto out ;
}
first_trk = blk_rq_pos ( req ) / DASD_RAW_SECTORS_PER_TRACK ;
last_trk = ( blk_rq_pos ( req ) + blk_rq_sectors ( req ) - 1 ) /
DASD_RAW_SECTORS_PER_TRACK ;
trkcount = last_trk - first_trk + 1 ;
first_offs = 0 ;
if ( rq_data_dir ( req ) = = READ )
cmd = DASD_ECKD_CCW_READ_TRACK ;
else if ( rq_data_dir ( req ) = = WRITE )
cmd = DASD_ECKD_CCW_WRITE_FULL_TRACK ;
else {
cqr = ERR_PTR ( - EINVAL ) ;
goto out ;
}
/*
* Raw track based I / O needs IDAWs for each page ,
* and not just for 64 bit addresses .
*/
cidaw = trkcount * DASD_RAW_BLOCK_PER_TRACK ;
/* 1x prefix + one read/write ccw per track */
cplength = 1 + trkcount ;
/*
* struct PFX_eckd_data has up to 2 byte as extended parameter
* this is needed for write full track and has to be mentioned
2011-03-30 22:57:33 -03:00
* separately
2011-01-05 12:48:06 +01:00
* add 8 instead of 2 to keep 8 byte boundary
*/
pfx_datasize = sizeof ( struct PFX_eckd_data ) + 8 ;
datasize = pfx_datasize + cidaw * sizeof ( unsigned long long ) ;
/* Allocate the ccw request. */
cqr = dasd_smalloc_request ( DASD_ECKD_MAGIC , cplength ,
datasize , startdev ) ;
if ( IS_ERR ( cqr ) )
goto out ;
ccw = cqr - > cpaddr ;
if ( prefix_LRE ( ccw + + , cqr - > data , first_trk , last_trk , cmd ,
basedev , startdev , 1 /* format */ , first_offs + 1 ,
trkcount , 0 , 0 ) = = - EAGAIN ) {
/* Clock not in sync and XRC is enabled.
* Try again later .
*/
dasd_sfree_request ( cqr , startdev ) ;
cqr = ERR_PTR ( - EAGAIN ) ;
goto out ;
}
idaws = ( unsigned long * ) ( cqr - > data + pfx_datasize ) ;
len_to_track_end = 0 ;
2013-08-16 15:57:32 +02:00
if ( start_padding_sectors ) {
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
ccw - > cmd_code = cmd ;
/* maximum 3390 track size */
ccw - > count = 57326 ;
/* 64k map to one track */
len_to_track_end = 65536 - start_padding_sectors * 512 ;
ccw - > cda = ( __u32 ) ( addr_t ) idaws ;
ccw - > flags | = CCW_FLAG_IDA ;
ccw - > flags | = CCW_FLAG_SLI ;
ccw + + ;
for ( sectors = 0 ; sectors < start_padding_sectors ; sectors + = 8 )
idaws = idal_create_words ( idaws , rawpadpage , PAGE_SIZE ) ;
}
2011-01-05 12:48:06 +01:00
rq_for_each_segment ( bv , req , iter ) {
2013-11-23 17:19:00 -08:00
dst = page_address ( bv . bv_page ) + bv . bv_offset ;
seg_len = bv . bv_len ;
2013-08-16 15:57:32 +02:00
if ( cmd = = DASD_ECKD_CCW_READ_TRACK )
memset ( dst , 0 , seg_len ) ;
2011-01-05 12:48:06 +01:00
if ( ! len_to_track_end ) {
ccw [ - 1 ] . flags | = CCW_FLAG_CC ;
ccw - > cmd_code = cmd ;
/* maximum 3390 track size */
ccw - > count = 57326 ;
/* 64k map to one track */
len_to_track_end = 65536 ;
ccw - > cda = ( __u32 ) ( addr_t ) idaws ;
ccw - > flags | = CCW_FLAG_IDA ;
ccw - > flags | = CCW_FLAG_SLI ;
ccw + + ;
}
len_to_track_end - = seg_len ;
idaws = idal_create_words ( idaws , dst , seg_len ) ;
}
2013-08-16 15:57:32 +02:00
for ( sectors = 0 ; sectors < end_padding_sectors ; sectors + = 8 )
idaws = idal_create_words ( idaws , rawpadpage , PAGE_SIZE ) ;
2011-01-05 12:48:06 +01:00
if ( blk_noretry_request ( req ) | |
block - > base - > features & DASD_FEATURE_FAILFAST )
set_bit ( DASD_CQR_FLAGS_FAILFAST , & cqr - > flags ) ;
cqr - > startdev = startdev ;
cqr - > memdev = startdev ;
cqr - > block = block ;
cqr - > expires = startdev - > default_expires * HZ ;
cqr - > lpm = startdev - > path_data . ppm ;
2013-01-30 09:26:12 +00:00
cqr - > retries = startdev - > default_retries ;
2013-01-30 09:49:40 +01:00
cqr - > buildclk = get_tod_clock ( ) ;
2011-01-05 12:48:06 +01:00
cqr - > status = DASD_CQR_FILLED ;
if ( IS_ERR ( cqr ) & & PTR_ERR ( cqr ) ! = - EAGAIN )
cqr = NULL ;
out :
return cqr ;
}
2005-04-16 15:20:36 -07:00
static int
dasd_eckd_free_cp ( struct dasd_ccw_req * cqr , struct request * req )
{
struct dasd_eckd_private * private ;
struct ccw1 * ccw ;
2007-09-25 12:35:59 +02:00
struct req_iterator iter ;
2013-11-23 17:19:00 -08:00
struct bio_vec bv ;
2005-04-16 15:20:36 -07:00
char * dst , * cda ;
unsigned int blksize , blk_per_trk , off ;
sector_t recid ;
2007-09-25 12:35:59 +02:00
int status ;
2005-04-16 15:20:36 -07:00
if ( ! dasd_page_cache )
goto out ;
2008-01-26 14:11:23 +01:00
private = ( struct dasd_eckd_private * ) cqr - > block - > base - > private ;
blksize = cqr - > block - > bp_block ;
2005-04-16 15:20:36 -07:00
blk_per_trk = recs_per_track ( & private - > rdc_data , 0 , blksize ) ;
2009-05-07 22:24:39 +09:00
recid = blk_rq_pos ( req ) > > cqr - > block - > s2b_shift ;
2005-04-16 15:20:36 -07:00
ccw = cqr - > cpaddr ;
/* Skip over define extent & locate record. */
ccw + + ;
if ( private - > uses_cdl = = 0 | | recid > 2 * blk_per_trk )
ccw + + ;
2007-09-25 12:35:59 +02:00
rq_for_each_segment ( bv , req , iter ) {
2013-11-23 17:19:00 -08:00
dst = page_address ( bv . bv_page ) + bv . bv_offset ;
for ( off = 0 ; off < bv . bv_len ; off + = blksize ) {
2005-04-16 15:20:36 -07:00
/* Skip locate record. */
if ( private - > uses_cdl & & recid < = 2 * blk_per_trk )
ccw + + ;
if ( dst ) {
if ( ccw - > flags & CCW_FLAG_IDA )
cda = * ( ( char * * ) ( ( addr_t ) ccw - > cda ) ) ;
else
cda = ( char * ) ( ( addr_t ) ccw - > cda ) ;
if ( dst ! = cda ) {
if ( rq_data_dir ( req ) = = READ )
2013-11-23 17:19:00 -08:00
memcpy ( dst , cda , bv . bv_len ) ;
2005-04-16 15:20:36 -07:00
kmem_cache_free ( dasd_page_cache ,
( void * ) ( ( addr_t ) cda & PAGE_MASK ) ) ;
}
dst = NULL ;
}
ccw + + ;
recid + + ;
}
}
out :
status = cqr - > status = = DASD_CQR_DONE ;
2008-01-26 14:11:23 +01:00
dasd_sfree_request ( cqr , cqr - > memdev ) ;
2005-04-16 15:20:36 -07:00
return status ;
}
2008-01-26 14:11:23 +01:00
/*
2009-03-26 15:23:48 +01:00
* Modify ccw / tcw in cqr so it can be started on a base device .
2008-01-26 14:11:23 +01:00
*
* Note that this is not enough to restart the cqr !
* Either reset cqr - > startdev as well ( summary unit check handling )
* or restart via separate cqr ( as in ERP handling ) .
*/
void dasd_eckd_reset_ccw_to_base_io ( struct dasd_ccw_req * cqr )
{
struct ccw1 * ccw ;
struct PFX_eckd_data * pfxdata ;
2009-03-26 15:23:48 +01:00
struct tcw * tcw ;
struct tccb * tccb ;
struct dcw * dcw ;
if ( cqr - > cpmode = = 1 ) {
tcw = cqr - > cpaddr ;
tccb = tcw_get_tccb ( tcw ) ;
dcw = ( struct dcw * ) & tccb - > tca [ 0 ] ;
pfxdata = ( struct PFX_eckd_data * ) & dcw - > cd [ 0 ] ;
2008-01-26 14:11:23 +01:00
pfxdata - > validity . verify_base = 0 ;
pfxdata - > validity . hyper_pav = 0 ;
2009-03-26 15:23:48 +01:00
} else {
ccw = cqr - > cpaddr ;
pfxdata = cqr - > data ;
if ( ccw - > cmd_code = = DASD_ECKD_CCW_PFX ) {
pfxdata - > validity . verify_base = 0 ;
pfxdata - > validity . hyper_pav = 0 ;
}
2008-01-26 14:11:23 +01:00
}
}
# define DASD_ECKD_CHANQ_MAX_SIZE 4
static struct dasd_ccw_req * dasd_eckd_build_alias_cp ( struct dasd_device * base ,
struct dasd_block * block ,
struct request * req )
{
struct dasd_eckd_private * private ;
struct dasd_device * startdev ;
unsigned long flags ;
struct dasd_ccw_req * cqr ;
startdev = dasd_alias_get_start_dev ( base ) ;
if ( ! startdev )
startdev = base ;
private = ( struct dasd_eckd_private * ) startdev - > private ;
if ( private - > count > = DASD_ECKD_CHANQ_MAX_SIZE )
return ERR_PTR ( - EBUSY ) ;
spin_lock_irqsave ( get_ccwdev_lock ( startdev - > cdev ) , flags ) ;
private - > count + + ;
2011-01-05 12:48:06 +01:00
if ( ( base - > features & DASD_FEATURE_USERAW ) )
cqr = dasd_raw_build_cp ( startdev , block , req ) ;
else
cqr = dasd_eckd_build_cp ( startdev , block , req ) ;
2008-01-26 14:11:23 +01:00
if ( IS_ERR ( cqr ) )
private - > count - - ;
spin_unlock_irqrestore ( get_ccwdev_lock ( startdev - > cdev ) , flags ) ;
return cqr ;
}
static int dasd_eckd_free_alias_cp ( struct dasd_ccw_req * cqr ,
struct request * req )
{
struct dasd_eckd_private * private ;
unsigned long flags ;
spin_lock_irqsave ( get_ccwdev_lock ( cqr - > memdev - > cdev ) , flags ) ;
private = ( struct dasd_eckd_private * ) cqr - > memdev - > private ;
private - > count - - ;
spin_unlock_irqrestore ( get_ccwdev_lock ( cqr - > memdev - > cdev ) , flags ) ;
return dasd_eckd_free_cp ( cqr , req ) ;
}
2005-04-16 15:20:36 -07:00
static int
dasd_eckd_fill_info ( struct dasd_device * device ,
struct dasd_information2_t * info )
{
struct dasd_eckd_private * private ;
private = ( struct dasd_eckd_private * ) device - > private ;
info - > label_block = 2 ;
info - > FBA_layout = private - > uses_cdl ? 0 : 1 ;
info - > format = private - > uses_cdl ? DASD_FORMAT_CDL : DASD_FORMAT_LDL ;
info - > characteristics_size = sizeof ( struct dasd_eckd_characteristics ) ;
memcpy ( info - > characteristics , & private - > rdc_data ,
sizeof ( struct dasd_eckd_characteristics ) ) ;
2008-08-01 16:39:09 +02:00
info - > confdata_size = min ( ( unsigned long ) private - > conf_len ,
sizeof ( info - > configuration_data ) ) ;
memcpy ( info - > configuration_data , private - > conf_data ,
info - > confdata_size ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
/*
* SECTION : ioctl functions for eckd devices .
*/
/*
* Release device ioctl .
2006-06-29 14:58:12 +02:00
* Buils a channel programm to releases a prior reserved
2005-04-16 15:20:36 -07:00
* ( see dasd_eckd_reserve ) device .
*/
static int
2006-03-24 03:15:20 -08:00
dasd_eckd_release ( struct dasd_device * device )
2005-04-16 15:20:36 -07:00
{
struct dasd_ccw_req * cqr ;
int rc ;
2009-03-26 15:23:48 +01:00
struct ccw1 * ccw ;
2010-08-09 18:12:59 +02:00
int useglobal ;
2005-04-16 15:20:36 -07:00
if ( ! capable ( CAP_SYS_ADMIN ) )
return - EACCES ;
2010-08-09 18:12:59 +02:00
useglobal = 0 ;
2009-09-11 10:28:29 +02:00
cqr = dasd_smalloc_request ( DASD_ECKD_MAGIC , 1 , 32 , device ) ;
2005-04-16 15:20:36 -07:00
if ( IS_ERR ( cqr ) ) {
2010-08-09 18:12:59 +02:00
mutex_lock ( & dasd_reserve_mutex ) ;
useglobal = 1 ;
cqr = & dasd_reserve_req - > cqr ;
memset ( cqr , 0 , sizeof ( * cqr ) ) ;
memset ( & dasd_reserve_req - > ccw , 0 ,
sizeof ( dasd_reserve_req - > ccw ) ) ;
cqr - > cpaddr = & dasd_reserve_req - > ccw ;
cqr - > data = & dasd_reserve_req - > data ;
cqr - > magic = DASD_ECKD_MAGIC ;
2005-04-16 15:20:36 -07:00
}
2009-03-26 15:23:48 +01:00
ccw = cqr - > cpaddr ;
ccw - > cmd_code = DASD_ECKD_CCW_RELEASE ;
ccw - > flags | = CCW_FLAG_SLI ;
ccw - > count = 32 ;
ccw - > cda = ( __u32 ) ( addr_t ) cqr - > data ;
2008-01-26 14:11:23 +01:00
cqr - > startdev = device ;
cqr - > memdev = device ;
2005-04-16 15:20:36 -07:00
clear_bit ( DASD_CQR_FLAGS_USE_ERP , & cqr - > flags ) ;
2006-01-06 00:19:15 -08:00
set_bit ( DASD_CQR_FLAGS_FAILFAST , & cqr - > flags ) ;
2007-02-05 21:17:24 +01:00
cqr - > retries = 2 ; /* set retry counter to enable basic ERP */
2005-04-16 15:20:36 -07:00
cqr - > expires = 2 * HZ ;
2013-01-30 09:49:40 +01:00
cqr - > buildclk = get_tod_clock ( ) ;
2005-04-16 15:20:36 -07:00
cqr - > status = DASD_CQR_FILLED ;
rc = dasd_sleep_on_immediatly ( cqr ) ;
2011-01-05 12:48:04 +01:00
if ( ! rc )
clear_bit ( DASD_FLAG_IS_RESERVED , & device - > flags ) ;
2005-04-16 15:20:36 -07:00
2010-08-09 18:12:59 +02:00
if ( useglobal )
mutex_unlock ( & dasd_reserve_mutex ) ;
else
dasd_sfree_request ( cqr , cqr - > memdev ) ;
2005-04-16 15:20:36 -07:00
return rc ;
}
/*
* Reserve device ioctl .
* Options are set to ' synchronous wait for interrupt ' and
2006-06-29 14:58:12 +02:00
* ' timeout the request ' . This leads to a terminate IO if
* the interrupt is outstanding for a certain time .
2005-04-16 15:20:36 -07:00
*/
static int
2006-03-24 03:15:20 -08:00
dasd_eckd_reserve ( struct dasd_device * device )
2005-04-16 15:20:36 -07:00
{
struct dasd_ccw_req * cqr ;
int rc ;
2009-03-26 15:23:48 +01:00
struct ccw1 * ccw ;
2010-08-09 18:12:59 +02:00
int useglobal ;
2005-04-16 15:20:36 -07:00
if ( ! capable ( CAP_SYS_ADMIN ) )
return - EACCES ;
2010-08-09 18:12:59 +02:00
useglobal = 0 ;
2009-09-11 10:28:29 +02:00
cqr = dasd_smalloc_request ( DASD_ECKD_MAGIC , 1 , 32 , device ) ;
2005-04-16 15:20:36 -07:00
if ( IS_ERR ( cqr ) ) {
2010-08-09 18:12:59 +02:00
mutex_lock ( & dasd_reserve_mutex ) ;
useglobal = 1 ;
cqr = & dasd_reserve_req - > cqr ;
memset ( cqr , 0 , sizeof ( * cqr ) ) ;
memset ( & dasd_reserve_req - > ccw , 0 ,
sizeof ( dasd_reserve_req - > ccw ) ) ;
cqr - > cpaddr = & dasd_reserve_req - > ccw ;
cqr - > data = & dasd_reserve_req - > data ;
cqr - > magic = DASD_ECKD_MAGIC ;
2005-04-16 15:20:36 -07:00
}
2009-03-26 15:23:48 +01:00
ccw = cqr - > cpaddr ;
ccw - > cmd_code = DASD_ECKD_CCW_RESERVE ;
ccw - > flags | = CCW_FLAG_SLI ;
ccw - > count = 32 ;
ccw - > cda = ( __u32 ) ( addr_t ) cqr - > data ;
2008-01-26 14:11:23 +01:00
cqr - > startdev = device ;
cqr - > memdev = device ;
2005-04-16 15:20:36 -07:00
clear_bit ( DASD_CQR_FLAGS_USE_ERP , & cqr - > flags ) ;
2006-01-06 00:19:15 -08:00
set_bit ( DASD_CQR_FLAGS_FAILFAST , & cqr - > flags ) ;
2007-02-05 21:17:24 +01:00
cqr - > retries = 2 ; /* set retry counter to enable basic ERP */
2005-04-16 15:20:36 -07:00
cqr - > expires = 2 * HZ ;
2013-01-30 09:49:40 +01:00
cqr - > buildclk = get_tod_clock ( ) ;
2005-04-16 15:20:36 -07:00
cqr - > status = DASD_CQR_FILLED ;
rc = dasd_sleep_on_immediatly ( cqr ) ;
2011-01-05 12:48:04 +01:00
if ( ! rc )
set_bit ( DASD_FLAG_IS_RESERVED , & device - > flags ) ;
2005-04-16 15:20:36 -07:00
2010-08-09 18:12:59 +02:00
if ( useglobal )
mutex_unlock ( & dasd_reserve_mutex ) ;
else
dasd_sfree_request ( cqr , cqr - > memdev ) ;
2005-04-16 15:20:36 -07:00
return rc ;
}
/*
* Steal lock ioctl - unconditional reserve device .
2006-06-29 14:58:12 +02:00
* Buils a channel programm to break a device ' s reservation .
2005-04-16 15:20:36 -07:00
* ( unconditional reserve )
*/
static int
2006-03-24 03:15:20 -08:00
dasd_eckd_steal_lock ( struct dasd_device * device )
2005-04-16 15:20:36 -07:00
{
struct dasd_ccw_req * cqr ;
int rc ;
2009-03-26 15:23:48 +01:00
struct ccw1 * ccw ;
2010-08-09 18:12:59 +02:00
int useglobal ;
2005-04-16 15:20:36 -07:00
if ( ! capable ( CAP_SYS_ADMIN ) )
return - EACCES ;
2010-08-09 18:12:59 +02:00
useglobal = 0 ;
2009-09-11 10:28:29 +02:00
cqr = dasd_smalloc_request ( DASD_ECKD_MAGIC , 1 , 32 , device ) ;
2005-04-16 15:20:36 -07:00
if ( IS_ERR ( cqr ) ) {
2010-08-09 18:12:59 +02:00
mutex_lock ( & dasd_reserve_mutex ) ;
useglobal = 1 ;
cqr = & dasd_reserve_req - > cqr ;
memset ( cqr , 0 , sizeof ( * cqr ) ) ;
memset ( & dasd_reserve_req - > ccw , 0 ,
sizeof ( dasd_reserve_req - > ccw ) ) ;
cqr - > cpaddr = & dasd_reserve_req - > ccw ;
cqr - > data = & dasd_reserve_req - > data ;
cqr - > magic = DASD_ECKD_MAGIC ;
2005-04-16 15:20:36 -07:00
}
2009-03-26 15:23:48 +01:00
ccw = cqr - > cpaddr ;
ccw - > cmd_code = DASD_ECKD_CCW_SLCK ;
ccw - > flags | = CCW_FLAG_SLI ;
ccw - > count = 32 ;
ccw - > cda = ( __u32 ) ( addr_t ) cqr - > data ;
2008-01-26 14:11:23 +01:00
cqr - > startdev = device ;
cqr - > memdev = device ;
2005-04-16 15:20:36 -07:00
clear_bit ( DASD_CQR_FLAGS_USE_ERP , & cqr - > flags ) ;
2006-01-06 00:19:15 -08:00
set_bit ( DASD_CQR_FLAGS_FAILFAST , & cqr - > flags ) ;
2007-02-05 21:17:24 +01:00
cqr - > retries = 2 ; /* set retry counter to enable basic ERP */
2005-04-16 15:20:36 -07:00
cqr - > expires = 2 * HZ ;
2013-01-30 09:49:40 +01:00
cqr - > buildclk = get_tod_clock ( ) ;
2005-04-16 15:20:36 -07:00
cqr - > status = DASD_CQR_FILLED ;
rc = dasd_sleep_on_immediatly ( cqr ) ;
2011-01-05 12:48:04 +01:00
if ( ! rc )
set_bit ( DASD_FLAG_IS_RESERVED , & device - > flags ) ;
2005-04-16 15:20:36 -07:00
2010-08-09 18:12:59 +02:00
if ( useglobal )
mutex_unlock ( & dasd_reserve_mutex ) ;
else
dasd_sfree_request ( cqr , cqr - > memdev ) ;
2005-04-16 15:20:36 -07:00
return rc ;
}
2010-10-29 16:50:43 +02:00
/*
* SNID - Sense Path Group ID
* This ioctl may be used in situations where I / O is stalled due to
* a reserve , so if the normal dasd_smalloc_request fails , we use the
* preallocated dasd_reserve_req .
*/
static int dasd_eckd_snid ( struct dasd_device * device ,
void __user * argp )
{
struct dasd_ccw_req * cqr ;
int rc ;
struct ccw1 * ccw ;
int useglobal ;
struct dasd_snid_ioctl_data usrparm ;
if ( ! capable ( CAP_SYS_ADMIN ) )
return - EACCES ;
if ( copy_from_user ( & usrparm , argp , sizeof ( usrparm ) ) )
return - EFAULT ;
useglobal = 0 ;
cqr = dasd_smalloc_request ( DASD_ECKD_MAGIC , 1 ,
sizeof ( struct dasd_snid_data ) , device ) ;
if ( IS_ERR ( cqr ) ) {
mutex_lock ( & dasd_reserve_mutex ) ;
useglobal = 1 ;
cqr = & dasd_reserve_req - > cqr ;
memset ( cqr , 0 , sizeof ( * cqr ) ) ;
memset ( & dasd_reserve_req - > ccw , 0 ,
sizeof ( dasd_reserve_req - > ccw ) ) ;
cqr - > cpaddr = & dasd_reserve_req - > ccw ;
cqr - > data = & dasd_reserve_req - > data ;
cqr - > magic = DASD_ECKD_MAGIC ;
}
ccw = cqr - > cpaddr ;
ccw - > cmd_code = DASD_ECKD_CCW_SNID ;
ccw - > flags | = CCW_FLAG_SLI ;
ccw - > count = 12 ;
ccw - > cda = ( __u32 ) ( addr_t ) cqr - > data ;
cqr - > startdev = device ;
cqr - > memdev = device ;
clear_bit ( DASD_CQR_FLAGS_USE_ERP , & cqr - > flags ) ;
set_bit ( DASD_CQR_FLAGS_FAILFAST , & cqr - > flags ) ;
2011-01-05 12:48:04 +01:00
set_bit ( DASD_CQR_ALLOW_SLOCK , & cqr - > flags ) ;
2010-10-29 16:50:43 +02:00
cqr - > retries = 5 ;
cqr - > expires = 10 * HZ ;
2013-01-30 09:49:40 +01:00
cqr - > buildclk = get_tod_clock ( ) ;
2010-10-29 16:50:43 +02:00
cqr - > status = DASD_CQR_FILLED ;
cqr - > lpm = usrparm . path_mask ;
rc = dasd_sleep_on_immediatly ( cqr ) ;
/* verify that I/O processing didn't modify the path mask */
if ( ! rc & & usrparm . path_mask & & ( cqr - > lpm ! = usrparm . path_mask ) )
rc = - EIO ;
if ( ! rc ) {
usrparm . data = * ( ( struct dasd_snid_data * ) cqr - > data ) ;
if ( copy_to_user ( argp , & usrparm , sizeof ( usrparm ) ) )
rc = - EFAULT ;
}
if ( useglobal )
mutex_unlock ( & dasd_reserve_mutex ) ;
else
dasd_sfree_request ( cqr , cqr - > memdev ) ;
return rc ;
}
2005-04-16 15:20:36 -07:00
/*
* Read performance statistics
*/
static int
2006-03-24 03:15:20 -08:00
dasd_eckd_performance ( struct dasd_device * device , void __user * argp )
2005-04-16 15:20:36 -07:00
{
struct dasd_psf_prssd_data * prssdp ;
struct dasd_rssd_perf_stats_t * stats ;
struct dasd_ccw_req * cqr ;
struct ccw1 * ccw ;
int rc ;
2009-09-11 10:28:29 +02:00
cqr = dasd_smalloc_request ( DASD_ECKD_MAGIC , 1 /* PSF */ + 1 /* RSSD */ ,
2008-01-26 14:11:23 +01:00
( sizeof ( struct dasd_psf_prssd_data ) +
sizeof ( struct dasd_rssd_perf_stats_t ) ) ,
2005-04-16 15:20:36 -07:00
device ) ;
if ( IS_ERR ( cqr ) ) {
2009-03-26 15:23:49 +01:00
DBF_DEV_EVENT ( DBF_WARNING , device , " %s " ,
2005-04-16 15:20:36 -07:00
" Could not allocate initialization request " ) ;
return PTR_ERR ( cqr ) ;
}
2008-01-26 14:11:23 +01:00
cqr - > startdev = device ;
cqr - > memdev = device ;
2005-04-16 15:20:36 -07:00
cqr - > retries = 0 ;
2009-12-07 12:51:51 +01:00
clear_bit ( DASD_CQR_FLAGS_USE_ERP , & cqr - > flags ) ;
2005-04-16 15:20:36 -07:00
cqr - > expires = 10 * HZ ;
/* Prepare for Read Subsystem Data */
prssdp = ( struct dasd_psf_prssd_data * ) cqr - > data ;
2008-01-26 14:11:23 +01:00
memset ( prssdp , 0 , sizeof ( struct dasd_psf_prssd_data ) ) ;
2005-04-16 15:20:36 -07:00
prssdp - > order = PSF_ORDER_PRSSD ;
2008-01-26 14:11:20 +01:00
prssdp - > suborder = 0x01 ; /* Performance Statistics */
2005-04-16 15:20:36 -07:00
prssdp - > varies [ 1 ] = 0x01 ; /* Perf Statistics for the Subsystem */
ccw = cqr - > cpaddr ;
ccw - > cmd_code = DASD_ECKD_CCW_PSF ;
2008-01-26 14:11:23 +01:00
ccw - > count = sizeof ( struct dasd_psf_prssd_data ) ;
2005-04-16 15:20:36 -07:00
ccw - > flags | = CCW_FLAG_CC ;
ccw - > cda = ( __u32 ) ( addr_t ) prssdp ;
/* Read Subsystem Data - Performance Statistics */
stats = ( struct dasd_rssd_perf_stats_t * ) ( prssdp + 1 ) ;
2008-01-26 14:11:23 +01:00
memset ( stats , 0 , sizeof ( struct dasd_rssd_perf_stats_t ) ) ;
2005-04-16 15:20:36 -07:00
ccw + + ;
ccw - > cmd_code = DASD_ECKD_CCW_RSSD ;
2008-01-26 14:11:23 +01:00
ccw - > count = sizeof ( struct dasd_rssd_perf_stats_t ) ;
2005-04-16 15:20:36 -07:00
ccw - > cda = ( __u32 ) ( addr_t ) stats ;
2013-01-30 09:49:40 +01:00
cqr - > buildclk = get_tod_clock ( ) ;
2005-04-16 15:20:36 -07:00
cqr - > status = DASD_CQR_FILLED ;
rc = dasd_sleep_on ( cqr ) ;
if ( rc = = 0 ) {
prssdp = ( struct dasd_psf_prssd_data * ) cqr - > data ;
stats = ( struct dasd_rssd_perf_stats_t * ) ( prssdp + 1 ) ;
2006-03-24 03:15:20 -08:00
if ( copy_to_user ( argp , stats ,
sizeof ( struct dasd_rssd_perf_stats_t ) ) )
rc = - EFAULT ;
2005-04-16 15:20:36 -07:00
}
2008-01-26 14:11:23 +01:00
dasd_sfree_request ( cqr , cqr - > memdev ) ;
2005-04-16 15:20:36 -07:00
return rc ;
}
/*
* Get attributes ( cache operations )
* Returnes the cache attributes used in Define Extend ( DE ) .
*/
static int
2006-03-24 03:15:20 -08:00
dasd_eckd_get_attrib ( struct dasd_device * device , void __user * argp )
2005-04-16 15:20:36 -07:00
{
2006-03-24 03:15:20 -08:00
struct dasd_eckd_private * private =
( struct dasd_eckd_private * ) device - > private ;
struct attrib_data_t attrib = private - > attrib ;
2005-04-16 15:20:36 -07:00
int rc ;
if ( ! capable ( CAP_SYS_ADMIN ) )
return - EACCES ;
2006-03-24 03:15:20 -08:00
if ( ! argp )
2005-04-16 15:20:36 -07:00
return - EINVAL ;
2006-03-24 03:15:20 -08:00
rc = 0 ;
if ( copy_to_user ( argp , ( long * ) & attrib ,
2008-01-26 14:11:23 +01:00
sizeof ( struct attrib_data_t ) ) )
2006-03-24 03:15:20 -08:00
rc = - EFAULT ;
2005-04-16 15:20:36 -07:00
return rc ;
}
/*
* Set attributes ( cache operations )
* Stores the attributes for cache operation to be used in Define Extend ( DE ) .
*/
static int
2006-03-24 03:15:20 -08:00
dasd_eckd_set_attrib ( struct dasd_device * device , void __user * argp )
2005-04-16 15:20:36 -07:00
{
2006-03-24 03:15:20 -08:00
struct dasd_eckd_private * private =
( struct dasd_eckd_private * ) device - > private ;
2005-04-16 15:20:36 -07:00
struct attrib_data_t attrib ;
if ( ! capable ( CAP_SYS_ADMIN ) )
return - EACCES ;
2006-03-24 03:15:20 -08:00
if ( ! argp )
2005-04-16 15:20:36 -07:00
return - EINVAL ;
2006-03-24 03:15:20 -08:00
if ( copy_from_user ( & attrib , argp , sizeof ( struct attrib_data_t ) ) )
2005-04-16 15:20:36 -07:00
return - EFAULT ;
private - > attrib = attrib ;
2009-03-26 15:23:49 +01:00
dev_info ( & device - > cdev - > dev ,
" The DASD cache mode was set to %x (%i cylinder prestage) \n " ,
private - > attrib . operation , private - > attrib . nr_cyl ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2008-10-10 21:33:25 +02:00
/*
* Issue syscall I / O to EMC Symmetrix array .
* CCWs are PSF and RSSD
*/
static int dasd_symm_io ( struct dasd_device * device , void __user * argp )
{
struct dasd_symmio_parms usrparm ;
char * psf_data , * rssd_result ;
struct dasd_ccw_req * cqr ;
struct ccw1 * ccw ;
2010-03-08 12:25:16 +01:00
char psf0 , psf1 ;
2008-10-10 21:33:25 +02:00
int rc ;
2010-03-08 12:25:16 +01:00
if ( ! capable ( CAP_SYS_ADMIN ) & & ! capable ( CAP_SYS_RAWIO ) )
return - EACCES ;
psf0 = psf1 = 0 ;
2008-10-10 21:33:25 +02:00
/* Copy parms from caller */
rc = - EFAULT ;
if ( copy_from_user ( & usrparm , argp , sizeof ( usrparm ) ) )
goto out ;
2015-08-13 13:26:49 +02:00
if ( is_compat_task ( ) ) {
2010-01-13 20:44:40 +01:00
/* Make sure pointers are sane even on 31 bit. */
2008-10-10 21:33:25 +02:00
rc = - EINVAL ;
2010-01-13 20:44:40 +01:00
if ( ( usrparm . psf_data > > 32 ) ! = 0 )
goto out ;
if ( ( usrparm . rssd_result > > 32 ) ! = 0 )
goto out ;
usrparm . psf_data & = 0x7fffffffULL ;
usrparm . rssd_result & = 0x7fffffffULL ;
2008-10-10 21:33:25 +02:00
}
/* alloc I/O data area */
psf_data = kzalloc ( usrparm . psf_data_len , GFP_KERNEL | GFP_DMA ) ;
rssd_result = kzalloc ( usrparm . rssd_result_len , GFP_KERNEL | GFP_DMA ) ;
if ( ! psf_data | | ! rssd_result ) {
rc = - ENOMEM ;
goto out_free ;
}
/* get syscall header from user space */
rc = - EFAULT ;
if ( copy_from_user ( psf_data ,
( void __user * ) ( unsigned long ) usrparm . psf_data ,
usrparm . psf_data_len ) )
goto out_free ;
2010-03-08 12:25:16 +01:00
psf0 = psf_data [ 0 ] ;
psf1 = psf_data [ 1 ] ;
2008-10-10 21:33:25 +02:00
/* setup CCWs for PSF + RSSD */
2009-09-11 10:28:29 +02:00
cqr = dasd_smalloc_request ( DASD_ECKD_MAGIC , 2 , 0 , device ) ;
2008-10-10 21:33:25 +02:00
if ( IS_ERR ( cqr ) ) {
2009-03-26 15:23:49 +01:00
DBF_DEV_EVENT ( DBF_WARNING , device , " %s " ,
2008-10-10 21:33:25 +02:00
" Could not allocate initialization request " ) ;
rc = PTR_ERR ( cqr ) ;
goto out_free ;
}
cqr - > startdev = device ;
cqr - > memdev = device ;
cqr - > retries = 3 ;
cqr - > expires = 10 * HZ ;
2013-01-30 09:49:40 +01:00
cqr - > buildclk = get_tod_clock ( ) ;
2008-10-10 21:33:25 +02:00
cqr - > status = DASD_CQR_FILLED ;
/* Build the ccws */
ccw = cqr - > cpaddr ;
/* PSF ccw */
ccw - > cmd_code = DASD_ECKD_CCW_PSF ;
ccw - > count = usrparm . psf_data_len ;
ccw - > flags | = CCW_FLAG_CC ;
ccw - > cda = ( __u32 ) ( addr_t ) psf_data ;
ccw + + ;
/* RSSD ccw */
ccw - > cmd_code = DASD_ECKD_CCW_RSSD ;
ccw - > count = usrparm . rssd_result_len ;
ccw - > flags = CCW_FLAG_SLI ;
ccw - > cda = ( __u32 ) ( addr_t ) rssd_result ;
rc = dasd_sleep_on ( cqr ) ;
if ( rc )
goto out_sfree ;
rc = - EFAULT ;
if ( copy_to_user ( ( void __user * ) ( unsigned long ) usrparm . rssd_result ,
rssd_result , usrparm . rssd_result_len ) )
goto out_sfree ;
rc = 0 ;
out_sfree :
dasd_sfree_request ( cqr , cqr - > memdev ) ;
out_free :
kfree ( rssd_result ) ;
kfree ( psf_data ) ;
out :
2010-03-08 12:25:16 +01:00
DBF_DEV_EVENT ( DBF_WARNING , device ,
" Symmetrix ioctl (0x%02x 0x%02x): rc=%d " ,
( int ) psf0 , ( int ) psf1 , rc ) ;
2008-10-10 21:33:25 +02:00
return rc ;
}
2006-03-24 03:15:20 -08:00
static int
2008-01-26 14:11:23 +01:00
dasd_eckd_ioctl ( struct dasd_block * block , unsigned int cmd , void __user * argp )
2006-03-24 03:15:20 -08:00
{
2008-01-26 14:11:23 +01:00
struct dasd_device * device = block - > base ;
2006-03-24 03:15:20 -08:00
switch ( cmd ) {
case BIODASDGATTR :
return dasd_eckd_get_attrib ( device , argp ) ;
case BIODASDSATTR :
return dasd_eckd_set_attrib ( device , argp ) ;
case BIODASDPSRD :
return dasd_eckd_performance ( device , argp ) ;
case BIODASDRLSE :
return dasd_eckd_release ( device ) ;
case BIODASDRSRV :
return dasd_eckd_reserve ( device ) ;
case BIODASDSLCK :
return dasd_eckd_steal_lock ( device ) ;
2010-10-29 16:50:43 +02:00
case BIODASDSNID :
return dasd_eckd_snid ( device , argp ) ;
2008-10-10 21:33:25 +02:00
case BIODASDSYMMIO :
return dasd_symm_io ( device , argp ) ;
2006-03-24 03:15:20 -08:00
default :
2012-08-27 10:59:42 +02:00
return - ENOTTY ;
2006-03-24 03:15:20 -08:00
}
}
2006-06-29 14:57:52 +02:00
/*
* Dump the range of CCWs into ' page ' buffer
* and return number of printed chars .
*/
2007-02-05 21:18:53 +01:00
static int
2006-06-29 14:57:52 +02:00
dasd_eckd_dump_ccw_range ( struct ccw1 * from , struct ccw1 * to , char * page )
{
int len , count ;
char * datap ;
len = 0 ;
while ( from < = to ) {
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2006-06-29 14:57:52 +02:00
" CCW %p: %08X %08X DAT: " ,
from , ( ( int * ) from ) [ 0 ] , ( ( int * ) from ) [ 1 ] ) ;
/* get pointer to data (consider IDALs) */
if ( from - > flags & CCW_FLAG_IDA )
datap = ( char * ) * ( ( addr_t * ) ( addr_t ) from - > cda ) ;
else
datap = ( char * ) ( ( addr_t ) from - > cda ) ;
/* dump data (max 32 bytes) */
for ( count = 0 ; count < from - > count & & count < 32 ; count + + ) {
if ( count % 8 = = 0 ) len + = sprintf ( page + len , " " ) ;
if ( count % 4 = = 0 ) len + = sprintf ( page + len , " " ) ;
len + = sprintf ( page + len , " %02x " , datap [ count ] ) ;
}
len + = sprintf ( page + len , " \n " ) ;
from + + ;
}
return len ;
}
2009-03-26 15:23:49 +01:00
static void
2009-07-07 16:37:06 +02:00
dasd_eckd_dump_sense_dbf ( struct dasd_device * device , struct irb * irb ,
char * reason )
2009-03-26 15:23:49 +01:00
{
u64 * sense ;
2010-10-25 16:10:47 +02:00
u64 * stat ;
2009-07-07 16:37:06 +02:00
sense = ( u64 * ) dasd_get_sense ( irb ) ;
2010-10-25 16:10:47 +02:00
stat = ( u64 * ) & irb - > scsw ;
2009-03-26 15:23:49 +01:00
if ( sense ) {
2010-10-25 16:10:47 +02:00
DBF_DEV_EVENT ( DBF_EMERG , device , " %s: %016llx %08x : "
" %016llx %016llx %016llx %016llx " ,
reason , * stat , * ( ( u32 * ) ( stat + 1 ) ) ,
2010-10-25 16:10:24 +02:00
sense [ 0 ] , sense [ 1 ] , sense [ 2 ] , sense [ 3 ] ) ;
2009-03-26 15:23:49 +01:00
} else {
2010-10-25 16:10:47 +02:00
DBF_DEV_EVENT ( DBF_EMERG , device , " %s: %016llx %08x : %s " ,
reason , * stat , * ( ( u32 * ) ( stat + 1 ) ) ,
" NO VALID SENSE " ) ;
2009-03-26 15:23:49 +01:00
}
}
2005-04-16 15:20:36 -07:00
/*
* Print sense data and related channel program .
* Parts are printed because printk buffer is only 1024 bytes .
*/
2009-03-26 15:23:48 +01:00
static void dasd_eckd_dump_sense_ccw ( struct dasd_device * device ,
2008-01-26 14:11:23 +01:00
struct dasd_ccw_req * req , struct irb * irb )
2005-04-16 15:20:36 -07:00
{
char * page ;
2006-06-29 14:57:52 +02:00
struct ccw1 * first , * last , * fail , * from , * to ;
int len , sl , sct ;
2005-04-16 15:20:36 -07:00
page = ( char * ) get_zeroed_page ( GFP_ATOMIC ) ;
if ( page = = NULL ) {
2009-03-26 15:23:49 +01:00
DBF_DEV_EVENT ( DBF_WARNING , device , " %s " ,
" No memory to dump sense data \n " ) ;
2005-04-16 15:20:36 -07:00
return ;
}
2006-06-29 14:57:52 +02:00
/* dump the sense data */
2012-10-23 20:28:37 +02:00
len = sprintf ( page , PRINTK_HEADER
2005-04-16 15:20:36 -07:00
" I/O status report for device %s: \n " ,
2008-10-10 21:33:09 +02:00
dev_name ( & device - > cdev - > dev ) ) ;
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2010-10-25 16:10:47 +02:00
" in req: %p CC:%02X FC:%02X AC:%02X SC:%02X DS:%02X "
" CS:%02X RC:%d \n " ,
req , scsw_cc ( & irb - > scsw ) , scsw_fctl ( & irb - > scsw ) ,
scsw_actl ( & irb - > scsw ) , scsw_stctl ( & irb - > scsw ) ,
scsw_dstat ( & irb - > scsw ) , scsw_cstat ( & irb - > scsw ) ,
req ? req - > intrc : 0 ) ;
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2005-04-16 15:20:36 -07:00
" device %s: Failing CCW: %p \n " ,
2008-10-10 21:33:09 +02:00
dev_name ( & device - > cdev - > dev ) ,
2008-07-14 09:58:50 +02:00
( void * ) ( addr_t ) irb - > scsw . cmd . cpa ) ;
2005-04-16 15:20:36 -07:00
if ( irb - > esw . esw0 . erw . cons ) {
for ( sl = 0 ; sl < 4 ; sl + + ) {
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2005-04-16 15:20:36 -07:00
" Sense(hex) %2d-%2d: " ,
( 8 * sl ) , ( ( 8 * sl ) + 7 ) ) ;
for ( sct = 0 ; sct < 8 ; sct + + ) {
len + = sprintf ( page + len , " %02x " ,
irb - > ecw [ 8 * sl + sct ] ) ;
}
len + = sprintf ( page + len , " \n " ) ;
}
if ( irb - > ecw [ 27 ] & DASD_SENSE_BIT_0 ) {
/* 24 Byte Sense Data */
2012-10-23 20:28:37 +02:00
sprintf ( page + len , PRINTK_HEADER
2006-06-29 14:57:52 +02:00
" 24 Byte: %x MSG %x, "
" %s MSGb to SYSOP \n " ,
irb - > ecw [ 7 ] > > 4 , irb - > ecw [ 7 ] & 0x0f ,
irb - > ecw [ 1 ] & 0x10 ? " " : " no " ) ;
2005-04-16 15:20:36 -07:00
} else {
/* 32 Byte Sense Data */
2012-10-23 20:28:37 +02:00
sprintf ( page + len , PRINTK_HEADER
2006-06-29 14:57:52 +02:00
" 32 Byte: Format: %x "
" Exception class %x \n " ,
irb - > ecw [ 6 ] & 0x0f , irb - > ecw [ 22 ] > > 4 ) ;
2005-04-16 15:20:36 -07:00
}
} else {
2012-10-23 20:28:37 +02:00
sprintf ( page + len , PRINTK_HEADER
2006-06-29 14:57:52 +02:00
" SORRY - NO VALID SENSE AVAILABLE \n " ) ;
2005-04-16 15:20:36 -07:00
}
2012-10-23 20:28:37 +02:00
printk ( KERN_ERR " %s " , page ) ;
2006-06-29 14:57:52 +02:00
2008-01-26 14:11:23 +01:00
if ( req ) {
/* req == NULL for unsolicited interrupts */
/* dump the Channel Program (max 140 Bytes per line) */
/* Count CCW and print first CCWs (maximum 1024 % 140 = 7) */
first = req - > cpaddr ;
for ( last = first ; last - > flags & ( CCW_FLAG_CC | CCW_FLAG_DC ) ; last + + ) ;
to = min ( first + 6 , last ) ;
2012-10-23 20:28:37 +02:00
len = sprintf ( page , PRINTK_HEADER
2008-01-26 14:11:23 +01:00
" Related CP in req: %p \n " , req ) ;
dasd_eckd_dump_ccw_range ( first , to , page + len ) ;
2012-10-23 20:28:37 +02:00
printk ( KERN_ERR " %s " , page ) ;
2005-04-16 15:20:36 -07:00
2008-01-26 14:11:23 +01:00
/* print failing CCW area (maximum 4) */
/* scsw->cda is either valid or zero */
len = 0 ;
from = + + to ;
2008-07-14 09:58:50 +02:00
fail = ( struct ccw1 * ) ( addr_t )
irb - > scsw . cmd . cpa ; /* failing CCW */
2008-01-26 14:11:23 +01:00
if ( from < fail - 2 ) {
from = fail - 2 ; /* there is a gap - print header */
2012-10-23 20:28:37 +02:00
len + = sprintf ( page , PRINTK_HEADER " ...... \n " ) ;
2008-01-26 14:11:23 +01:00
}
to = min ( fail + 1 , last ) ;
len + = dasd_eckd_dump_ccw_range ( from , to , page + len ) ;
/* print last CCWs (maximum 2) */
from = max ( from , + + to ) ;
if ( from < last - 1 ) {
from = last - 1 ; /* there is a gap - print header */
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER " ...... \n " ) ;
2008-01-26 14:11:23 +01:00
}
len + = dasd_eckd_dump_ccw_range ( from , last , page + len ) ;
if ( len > 0 )
2012-10-23 20:28:37 +02:00
printk ( KERN_ERR " %s " , page ) ;
2005-04-16 15:20:36 -07:00
}
free_page ( ( unsigned long ) page ) ;
}
2009-03-26 15:23:48 +01:00
/*
* Print sense data from a tcw .
*/
static void dasd_eckd_dump_sense_tcw ( struct dasd_device * device ,
struct dasd_ccw_req * req , struct irb * irb )
{
char * page ;
int len , sl , sct , residual ;
struct tsb * tsb ;
2011-01-05 12:48:02 +01:00
u8 * sense , * rcq ;
2009-03-26 15:23:48 +01:00
page = ( char * ) get_zeroed_page ( GFP_ATOMIC ) ;
if ( page = = NULL ) {
2009-03-26 15:23:49 +01:00
DBF_DEV_EVENT ( DBF_WARNING , device , " %s " ,
2009-03-26 15:23:48 +01:00
" No memory to dump sense data " ) ;
return ;
}
/* dump the sense data */
2012-10-23 20:28:37 +02:00
len = sprintf ( page , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" I/O status report for device %s: \n " ,
dev_name ( & device - > cdev - > dev ) ) ;
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2010-10-25 16:10:47 +02:00
" in req: %p CC:%02X FC:%02X AC:%02X SC:%02X DS:%02X "
" CS:%02X fcxs:%02X schxs:%02X RC:%d \n " ,
req , scsw_cc ( & irb - > scsw ) , scsw_fctl ( & irb - > scsw ) ,
scsw_actl ( & irb - > scsw ) , scsw_stctl ( & irb - > scsw ) ,
scsw_dstat ( & irb - > scsw ) , scsw_cstat ( & irb - > scsw ) ,
irb - > scsw . tm . fcxs , irb - > scsw . tm . schxs ,
req ? req - > intrc : 0 ) ;
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" device %s: Failing TCW: %p \n " ,
dev_name ( & device - > cdev - > dev ) ,
( void * ) ( addr_t ) irb - > scsw . tm . tcw ) ;
tsb = NULL ;
sense = NULL ;
2010-10-25 16:10:47 +02:00
if ( irb - > scsw . tm . tcw & & ( irb - > scsw . tm . fcxs & 0x01 ) )
2009-03-26 15:23:48 +01:00
tsb = tcw_get_tsb (
( struct tcw * ) ( unsigned long ) irb - > scsw . tm . tcw ) ;
2010-03-24 11:49:54 +01:00
if ( tsb ) {
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" tsb->length %d \n " , tsb - > length ) ;
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" tsb->flags %x \n " , tsb - > flags ) ;
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" tsb->dcw_offset %d \n " , tsb - > dcw_offset ) ;
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" tsb->count %d \n " , tsb - > count ) ;
residual = tsb - > count - 28 ;
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" residual %d \n " , residual ) ;
switch ( tsb - > flags & 0x07 ) {
case 1 : /* tsa_iostat */
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" tsb->tsa.iostat.dev_time %d \n " ,
tsb - > tsa . iostat . dev_time ) ;
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" tsb->tsa.iostat.def_time %d \n " ,
tsb - > tsa . iostat . def_time ) ;
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" tsb->tsa.iostat.queue_time %d \n " ,
tsb - > tsa . iostat . queue_time ) ;
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" tsb->tsa.iostat.dev_busy_time %d \n " ,
tsb - > tsa . iostat . dev_busy_time ) ;
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" tsb->tsa.iostat.dev_act_time %d \n " ,
tsb - > tsa . iostat . dev_act_time ) ;
sense = tsb - > tsa . iostat . sense ;
break ;
case 2 : /* ts_ddpc */
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" tsb->tsa.ddpc.rc %d \n " , tsb - > tsa . ddpc . rc ) ;
2011-01-05 12:48:02 +01:00
for ( sl = 0 ; sl < 2 ; sl + + ) {
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2011-01-05 12:48:02 +01:00
" tsb->tsa.ddpc.rcq %2d-%2d: " ,
( 8 * sl ) , ( ( 8 * sl ) + 7 ) ) ;
rcq = tsb - > tsa . ddpc . rcq ;
2009-03-26 15:23:48 +01:00
for ( sct = 0 ; sct < 8 ; sct + + ) {
len + = sprintf ( page + len , " %02x " ,
2011-01-05 12:48:02 +01:00
rcq [ 8 * sl + sct ] ) ;
2009-03-26 15:23:48 +01:00
}
len + = sprintf ( page + len , " \n " ) ;
}
sense = tsb - > tsa . ddpc . sense ;
break ;
case 3 : /* tsa_intrg */
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
" tsb->tsa.intrg.: not supportet yet \n " ) ;
2009-03-26 15:23:48 +01:00
break ;
}
if ( sense ) {
for ( sl = 0 ; sl < 4 ; sl + + ) {
2012-10-23 20:28:37 +02:00
len + = sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" Sense(hex) %2d-%2d: " ,
( 8 * sl ) , ( ( 8 * sl ) + 7 ) ) ;
for ( sct = 0 ; sct < 8 ; sct + + ) {
len + = sprintf ( page + len , " %02x " ,
sense [ 8 * sl + sct ] ) ;
}
len + = sprintf ( page + len , " \n " ) ;
}
if ( sense [ 27 ] & DASD_SENSE_BIT_0 ) {
/* 24 Byte Sense Data */
2012-10-23 20:28:37 +02:00
sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" 24 Byte: %x MSG %x, "
" %s MSGb to SYSOP \n " ,
sense [ 7 ] > > 4 , sense [ 7 ] & 0x0f ,
sense [ 1 ] & 0x10 ? " " : " no " ) ;
} else {
/* 32 Byte Sense Data */
2012-10-23 20:28:37 +02:00
sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" 32 Byte: Format: %x "
" Exception class %x \n " ,
sense [ 6 ] & 0x0f , sense [ 22 ] > > 4 ) ;
}
} else {
2012-10-23 20:28:37 +02:00
sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" SORRY - NO VALID SENSE AVAILABLE \n " ) ;
}
} else {
2012-10-23 20:28:37 +02:00
sprintf ( page + len , PRINTK_HEADER
2009-03-26 15:23:48 +01:00
" SORRY - NO TSB DATA AVAILABLE \n " ) ;
}
2012-10-23 20:28:37 +02:00
printk ( KERN_ERR " %s " , page ) ;
2009-03-26 15:23:48 +01:00
free_page ( ( unsigned long ) page ) ;
}
static void dasd_eckd_dump_sense ( struct dasd_device * device ,
struct dasd_ccw_req * req , struct irb * irb )
{
2010-10-25 16:10:47 +02:00
if ( scsw_is_tm ( & irb - > scsw ) )
2009-03-26 15:23:48 +01:00
dasd_eckd_dump_sense_tcw ( device , req , irb ) ;
else
dasd_eckd_dump_sense_ccw ( device , req , irb ) ;
}
2010-05-17 10:00:10 +02:00
static int dasd_eckd_pm_freeze ( struct dasd_device * device )
2009-06-16 10:30:25 +02:00
{
/*
* the device should be disconnected from our LCU structure
* on restore we will reconnect it and reread LCU specific
* information like PAV support that might have changed
*/
dasd_alias_remove_device ( device ) ;
dasd_alias_disconnect_device_from_lcu ( device ) ;
return 0 ;
}
2010-05-17 10:00:10 +02:00
static int dasd_eckd_restore_device ( struct dasd_device * device )
2009-06-16 10:30:25 +02:00
{
struct dasd_eckd_private * private ;
2009-10-06 10:34:15 +02:00
struct dasd_eckd_characteristics temp_rdc_data ;
2012-01-18 18:03:40 +01:00
int rc ;
2009-06-16 10:30:25 +02:00
struct dasd_uid temp_uid ;
2009-10-14 12:43:46 +02:00
unsigned long flags ;
2013-02-19 09:30:05 +01:00
unsigned long cqr_flags = 0 ;
2009-06-16 10:30:25 +02:00
private = ( struct dasd_eckd_private * ) device - > private ;
/* Read Configuration Data */
2012-11-27 17:04:14 +01:00
dasd_eckd_read_conf ( device ) ;
2009-06-16 10:30:25 +02:00
2010-05-17 10:00:11 +02:00
dasd_eckd_get_uid ( device , & temp_uid ) ;
/* Generate device unique id */
rc = dasd_eckd_generate_uid ( device ) ;
spin_lock_irqsave ( get_ccwdev_lock ( device - > cdev ) , flags ) ;
2009-06-16 10:30:25 +02:00
if ( memcmp ( & private - > uid , & temp_uid , sizeof ( struct dasd_uid ) ) ! = 0 )
2009-10-14 12:43:46 +02:00
dev_err ( & device - > cdev - > dev , " The UID of the DASD has "
" changed \n " ) ;
2010-05-17 10:00:11 +02:00
spin_unlock_irqrestore ( get_ccwdev_lock ( device - > cdev ) , flags ) ;
2009-06-16 10:30:25 +02:00
if ( rc )
goto out_err ;
/* register lcu with alias handling, enable PAV if this is a new lcu */
2012-01-18 18:03:40 +01:00
rc = dasd_alias_make_device_known_to_lcu ( device ) ;
if ( rc )
return rc ;
2013-02-19 09:30:05 +01:00
set_bit ( DASD_CQR_FLAGS_FAILFAST , & cqr_flags ) ;
dasd_eckd_validate_server ( device , cqr_flags ) ;
2009-12-07 12:51:53 +01:00
/* RE-Read Configuration Data */
2012-11-27 17:04:14 +01:00
dasd_eckd_read_conf ( device ) ;
2009-06-16 10:30:25 +02:00
/* Read Feature Codes */
2009-09-22 22:58:52 +02:00
dasd_eckd_read_features ( device ) ;
2009-06-16 10:30:25 +02:00
/* Read Device Characteristics */
2009-09-11 10:28:29 +02:00
rc = dasd_generic_read_dev_chars ( device , DASD_ECKD_MAGIC ,
2009-10-06 10:34:15 +02:00
& temp_rdc_data , 64 ) ;
2009-06-16 10:30:25 +02:00
if ( rc ) {
2009-12-07 12:51:52 +01:00
DBF_EVENT_DEVID ( DBF_WARNING , device - > cdev ,
" Read device characteristic failed, rc=%d " , rc ) ;
2009-06-16 10:30:25 +02:00
goto out_err ;
}
2009-10-14 12:43:46 +02:00
spin_lock_irqsave ( get_ccwdev_lock ( device - > cdev ) , flags ) ;
2009-10-06 10:34:15 +02:00
memcpy ( & private - > rdc_data , & temp_rdc_data , sizeof ( temp_rdc_data ) ) ;
2009-10-14 12:43:46 +02:00
spin_unlock_irqrestore ( get_ccwdev_lock ( device - > cdev ) , flags ) ;
2009-06-16 10:30:25 +02:00
/* add device to alias management */
dasd_alias_add_device ( device ) ;
return 0 ;
out_err :
2009-06-22 12:08:17 +02:00
return - 1 ;
2009-06-16 10:30:25 +02:00
}
2010-05-17 10:00:10 +02:00
static int dasd_eckd_reload_device ( struct dasd_device * device )
{
struct dasd_eckd_private * private ;
int rc , old_base ;
2010-05-17 10:00:11 +02:00
char print_uid [ 60 ] ;
struct dasd_uid uid ;
unsigned long flags ;
2010-05-17 10:00:10 +02:00
private = ( struct dasd_eckd_private * ) device - > private ;
2010-05-17 10:00:11 +02:00
spin_lock_irqsave ( get_ccwdev_lock ( device - > cdev ) , flags ) ;
2010-05-17 10:00:10 +02:00
old_base = private - > uid . base_unit_addr ;
2010-05-17 10:00:11 +02:00
spin_unlock_irqrestore ( get_ccwdev_lock ( device - > cdev ) , flags ) ;
2010-05-17 10:00:10 +02:00
/* Read Configuration Data */
rc = dasd_eckd_read_conf ( device ) ;
if ( rc )
goto out_err ;
2010-05-17 10:00:11 +02:00
rc = dasd_eckd_generate_uid ( device ) ;
2010-05-17 10:00:10 +02:00
if ( rc )
goto out_err ;
/*
* update unit address configuration and
* add device to alias management
*/
dasd_alias_update_add_device ( device ) ;
2010-05-17 10:00:11 +02:00
dasd_eckd_get_uid ( device , & uid ) ;
if ( old_base ! = uid . base_unit_addr ) {
if ( strlen ( uid . vduit ) > 0 )
snprintf ( print_uid , sizeof ( print_uid ) ,
" %s.%s.%04x.%02x.%s " , uid . vendor , uid . serial ,
uid . ssid , uid . base_unit_addr , uid . vduit ) ;
2010-05-17 10:00:10 +02:00
else
2010-05-17 10:00:11 +02:00
snprintf ( print_uid , sizeof ( print_uid ) ,
" %s.%s.%04x.%02x " , uid . vendor , uid . serial ,
uid . ssid , uid . base_unit_addr ) ;
2010-05-17 10:00:10 +02:00
dev_info ( & device - > cdev - > dev ,
" An Alias device was reassigned to a new base device "
2010-05-17 10:00:11 +02:00
" with UID: %s \n " , print_uid ) ;
2010-05-17 10:00:10 +02:00
}
return 0 ;
out_err :
return - 1 ;
}
2014-10-01 14:39:47 +02:00
static int dasd_eckd_read_message_buffer ( struct dasd_device * device ,
struct dasd_rssd_messages * messages ,
__u8 lpum )
{
struct dasd_rssd_messages * message_buf ;
struct dasd_psf_prssd_data * prssdp ;
struct dasd_eckd_private * private ;
struct dasd_ccw_req * cqr ;
struct ccw1 * ccw ;
int rc ;
private = ( struct dasd_eckd_private * ) device - > private ;
cqr = dasd_smalloc_request ( DASD_ECKD_MAGIC , 1 /* PSF */ + 1 /* RSSD */ ,
( sizeof ( struct dasd_psf_prssd_data ) +
sizeof ( struct dasd_rssd_messages ) ) ,
device ) ;
if ( IS_ERR ( cqr ) ) {
DBF_EVENT_DEVID ( DBF_WARNING , device - > cdev , " %s " ,
" Could not allocate read message buffer request " ) ;
return PTR_ERR ( cqr ) ;
}
cqr - > startdev = device ;
cqr - > memdev = device ;
cqr - > block = NULL ;
cqr - > expires = 10 * HZ ;
set_bit ( DASD_CQR_VERIFY_PATH , & cqr - > flags ) ;
2015-08-07 13:20:16 +02:00
/* dasd_sleep_on_immediatly does not do complex error
* recovery so clear erp flag and set retry counter to
* do basic erp */
clear_bit ( DASD_CQR_FLAGS_USE_ERP , & cqr - > flags ) ;
cqr - > retries = 256 ;
2014-10-01 14:39:47 +02:00
/* Prepare for Read Subsystem Data */
prssdp = ( struct dasd_psf_prssd_data * ) cqr - > data ;
memset ( prssdp , 0 , sizeof ( struct dasd_psf_prssd_data ) ) ;
prssdp - > order = PSF_ORDER_PRSSD ;
prssdp - > suborder = 0x03 ; /* Message Buffer */
/* all other bytes of prssdp must be zero */
ccw = cqr - > cpaddr ;
ccw - > cmd_code = DASD_ECKD_CCW_PSF ;
ccw - > count = sizeof ( struct dasd_psf_prssd_data ) ;
ccw - > flags | = CCW_FLAG_CC ;
ccw - > flags | = CCW_FLAG_SLI ;
ccw - > cda = ( __u32 ) ( addr_t ) prssdp ;
/* Read Subsystem Data - message buffer */
message_buf = ( struct dasd_rssd_messages * ) ( prssdp + 1 ) ;
memset ( message_buf , 0 , sizeof ( struct dasd_rssd_messages ) ) ;
ccw + + ;
ccw - > cmd_code = DASD_ECKD_CCW_RSSD ;
ccw - > count = sizeof ( struct dasd_rssd_messages ) ;
ccw - > flags | = CCW_FLAG_SLI ;
ccw - > cda = ( __u32 ) ( addr_t ) message_buf ;
cqr - > buildclk = get_tod_clock ( ) ;
cqr - > status = DASD_CQR_FILLED ;
rc = dasd_sleep_on_immediatly ( cqr ) ;
if ( rc = = 0 ) {
prssdp = ( struct dasd_psf_prssd_data * ) cqr - > data ;
message_buf = ( struct dasd_rssd_messages * )
( prssdp + 1 ) ;
memcpy ( messages , message_buf ,
sizeof ( struct dasd_rssd_messages ) ) ;
} else
DBF_EVENT_DEVID ( DBF_WARNING , device - > cdev ,
" Reading messages failed with rc=%d \n "
, rc ) ;
dasd_sfree_request ( cqr , cqr - > memdev ) ;
return rc ;
}
/*
* Perform Subsystem Function - CUIR response
*/
static int
dasd_eckd_psf_cuir_response ( struct dasd_device * device , int response ,
__u32 message_id ,
struct channel_path_desc * desc ,
struct subchannel_id sch_id )
{
struct dasd_psf_cuir_response * psf_cuir ;
struct dasd_ccw_req * cqr ;
struct ccw1 * ccw ;
int rc ;
cqr = dasd_smalloc_request ( DASD_ECKD_MAGIC , 1 /* PSF */ ,
sizeof ( struct dasd_psf_cuir_response ) ,
device ) ;
if ( IS_ERR ( cqr ) ) {
DBF_DEV_EVENT ( DBF_WARNING , device , " %s " ,
" Could not allocate PSF-CUIR request " ) ;
return PTR_ERR ( cqr ) ;
}
psf_cuir = ( struct dasd_psf_cuir_response * ) cqr - > data ;
psf_cuir - > order = PSF_ORDER_CUIR_RESPONSE ;
psf_cuir - > cc = response ;
if ( desc )
psf_cuir - > chpid = desc - > chpid ;
psf_cuir - > message_id = message_id ;
psf_cuir - > cssid = sch_id . cssid ;
psf_cuir - > ssid = sch_id . ssid ;
ccw = cqr - > cpaddr ;
ccw - > cmd_code = DASD_ECKD_CCW_PSF ;
ccw - > cda = ( __u32 ) ( addr_t ) psf_cuir ;
2015-08-07 13:20:16 +02:00
ccw - > flags = CCW_FLAG_SLI ;
2014-10-01 14:39:47 +02:00
ccw - > count = sizeof ( struct dasd_psf_cuir_response ) ;
cqr - > startdev = device ;
cqr - > memdev = device ;
cqr - > block = NULL ;
cqr - > retries = 256 ;
cqr - > expires = 10 * HZ ;
cqr - > buildclk = get_tod_clock ( ) ;
cqr - > status = DASD_CQR_FILLED ;
2015-08-07 13:20:16 +02:00
set_bit ( DASD_CQR_VERIFY_PATH , & cqr - > flags ) ;
2014-10-01 14:39:47 +02:00
rc = dasd_sleep_on ( cqr ) ;
dasd_sfree_request ( cqr , cqr - > memdev ) ;
return rc ;
}
2015-08-07 13:20:16 +02:00
/*
* return configuration data that is referenced by record selector
* if a record selector is specified or per default return the
* conf_data pointer for the path specified by lpum
*/
static struct dasd_conf_data * dasd_eckd_get_ref_conf ( struct dasd_device * device ,
__u8 lpum ,
struct dasd_cuir_message * cuir )
2014-10-01 14:39:47 +02:00
{
2015-08-07 13:20:16 +02:00
struct dasd_eckd_private * private ;
struct dasd_conf_data * conf_data ;
int path , pos ;
2014-10-01 14:39:47 +02:00
2015-08-07 13:20:16 +02:00
private = ( struct dasd_eckd_private * ) device - > private ;
if ( cuir - > record_selector = = 0 )
goto out ;
for ( path = 0x80 , pos = 0 ; path ; path > > = 1 , pos + + ) {
conf_data = private - > path_conf_data [ pos ] ;
if ( conf_data - > gneq . record_selector = =
cuir - > record_selector )
return conf_data ;
2014-10-01 14:39:47 +02:00
}
2015-08-07 13:20:16 +02:00
out :
return private - > path_conf_data [ 8 - ffs ( lpum ) ] ;
2014-10-01 14:39:47 +02:00
}
/*
2015-08-07 13:20:16 +02:00
* This function determines the scope of a reconfiguration request by
* analysing the path and device selection data provided in the CUIR request .
* Returns a path mask containing CUIR affected paths for the give device .
*
* If the CUIR request does not contain the required information return the
* path mask of the path the attention message for the CUIR request was reveived
* on .
*/
static int dasd_eckd_cuir_scope ( struct dasd_device * device , __u8 lpum ,
struct dasd_cuir_message * cuir )
{
struct dasd_conf_data * ref_conf_data ;
unsigned long bitmask = 0 , mask = 0 ;
struct dasd_eckd_private * private ;
struct dasd_conf_data * conf_data ;
unsigned int pos , path ;
char * ref_gneq , * gneq ;
char * ref_ned , * ned ;
int tbcpm = 0 ;
/* if CUIR request does not specify the scope use the path
the attention message was presented on */
if ( ! cuir - > ned_map | |
! ( cuir - > neq_map [ 0 ] | cuir - > neq_map [ 1 ] | cuir - > neq_map [ 2 ] ) )
return lpum ;
private = ( struct dasd_eckd_private * ) device - > private ;
/* get reference conf data */
ref_conf_data = dasd_eckd_get_ref_conf ( device , lpum , cuir ) ;
/* reference ned is determined by ned_map field */
pos = 8 - ffs ( cuir - > ned_map ) ;
ref_ned = ( char * ) & ref_conf_data - > neds [ pos ] ;
ref_gneq = ( char * ) & ref_conf_data - > gneq ;
/* transfer 24 bit neq_map to mask */
mask = cuir - > neq_map [ 2 ] ;
mask | = cuir - > neq_map [ 1 ] < < 8 ;
mask | = cuir - > neq_map [ 0 ] < < 16 ;
for ( path = 0x80 ; path ; path > > = 1 ) {
/* initialise data per path */
bitmask = mask ;
pos = 8 - ffs ( path ) ;
conf_data = private - > path_conf_data [ pos ] ;
pos = 8 - ffs ( cuir - > ned_map ) ;
ned = ( char * ) & conf_data - > neds [ pos ] ;
/* compare reference ned and per path ned */
if ( memcmp ( ref_ned , ned , sizeof ( * ned ) ) ! = 0 )
continue ;
gneq = ( char * ) & conf_data - > gneq ;
/* compare reference gneq and per_path gneq under
24 bit mask where mask bit 0 equals byte 7 of
the gneq and mask bit 24 equals byte 31 */
while ( bitmask ) {
pos = ffs ( bitmask ) - 1 ;
if ( memcmp ( & ref_gneq [ 31 - pos ] , & gneq [ 31 - pos ] , 1 )
! = 0 )
break ;
clear_bit ( pos , & bitmask ) ;
}
if ( bitmask )
continue ;
/* device and path match the reference values
add path to CUIR scope */
tbcpm | = path ;
}
return tbcpm ;
}
static void dasd_eckd_cuir_notify_user ( struct dasd_device * device ,
unsigned long paths ,
struct subchannel_id sch_id , int action )
{
struct channel_path_desc * desc ;
int pos ;
while ( paths ) {
/* get position of bit in mask */
pos = ffs ( paths ) - 1 ;
/* get channel path descriptor from this position */
desc = ccw_device_get_chp_desc ( device - > cdev , 7 - pos ) ;
if ( action = = CUIR_QUIESCE )
pr_warn ( " Service on the storage server caused path "
" %x.%02x to go offline " , sch_id . cssid ,
desc ? desc - > chpid : 0 ) ;
else if ( action = = CUIR_RESUME )
pr_info ( " Path %x.%02x is back online after service "
" on the storage server " , sch_id . cssid ,
desc ? desc - > chpid : 0 ) ;
kfree ( desc ) ;
clear_bit ( pos , & paths ) ;
}
}
static int dasd_eckd_cuir_remove_path ( struct dasd_device * device , __u8 lpum ,
struct dasd_cuir_message * cuir )
{
unsigned long tbcpm ;
tbcpm = dasd_eckd_cuir_scope ( device , lpum , cuir ) ;
/* nothing to do if path is not in use */
if ( ! ( device - > path_data . opm & tbcpm ) )
return 0 ;
if ( ! ( device - > path_data . opm & ~ tbcpm ) ) {
/* no path would be left if the CUIR action is taken
return error */
return - EINVAL ;
}
/* remove device from operational path mask */
device - > path_data . opm & = ~ tbcpm ;
device - > path_data . cuirpm | = tbcpm ;
return tbcpm ;
}
/*
* walk through all devices and build a path mask to quiesce them
* return an error if the last path to a device would be removed
2014-10-01 14:39:47 +02:00
*
* if only part of the devices are quiesced and an error
* occurs no onlining necessary , the storage server will
* notify the already set offline devices again
*/
static int dasd_eckd_cuir_quiesce ( struct dasd_device * device , __u8 lpum ,
2015-08-07 13:20:16 +02:00
struct subchannel_id sch_id ,
struct dasd_cuir_message * cuir )
2014-10-01 14:39:47 +02:00
{
struct alias_pav_group * pavgroup , * tempgroup ;
struct dasd_eckd_private * private ;
struct dasd_device * dev , * n ;
2015-08-07 13:20:16 +02:00
unsigned long paths = 0 ;
unsigned long flags ;
int tbcpm ;
2014-10-01 14:39:47 +02:00
private = ( struct dasd_eckd_private * ) device - > private ;
/* active devices */
2015-08-07 13:20:16 +02:00
list_for_each_entry_safe ( dev , n , & private - > lcu - > active_devices ,
2014-10-01 14:39:47 +02:00
alias_list ) {
2015-08-07 13:20:16 +02:00
spin_lock_irqsave ( get_ccwdev_lock ( dev - > cdev ) , flags ) ;
tbcpm = dasd_eckd_cuir_remove_path ( dev , lpum , cuir ) ;
spin_unlock_irqrestore ( get_ccwdev_lock ( dev - > cdev ) , flags ) ;
if ( tbcpm < 0 )
goto out_err ;
paths | = tbcpm ;
2014-10-01 14:39:47 +02:00
}
/* inactive devices */
2015-08-07 13:20:16 +02:00
list_for_each_entry_safe ( dev , n , & private - > lcu - > inactive_devices ,
2014-10-01 14:39:47 +02:00
alias_list ) {
2015-08-07 13:20:16 +02:00
spin_lock_irqsave ( get_ccwdev_lock ( dev - > cdev ) , flags ) ;
tbcpm = dasd_eckd_cuir_remove_path ( dev , lpum , cuir ) ;
spin_unlock_irqrestore ( get_ccwdev_lock ( dev - > cdev ) , flags ) ;
if ( tbcpm < 0 )
goto out_err ;
paths | = tbcpm ;
2014-10-01 14:39:47 +02:00
}
/* devices in PAV groups */
list_for_each_entry_safe ( pavgroup , tempgroup ,
& private - > lcu - > grouplist , group ) {
list_for_each_entry_safe ( dev , n , & pavgroup - > baselist ,
alias_list ) {
2015-08-07 13:20:16 +02:00
spin_lock_irqsave ( get_ccwdev_lock ( dev - > cdev ) , flags ) ;
tbcpm = dasd_eckd_cuir_remove_path ( dev , lpum , cuir ) ;
spin_unlock_irqrestore (
get_ccwdev_lock ( dev - > cdev ) , flags ) ;
if ( tbcpm < 0 )
goto out_err ;
paths | = tbcpm ;
2014-10-01 14:39:47 +02:00
}
list_for_each_entry_safe ( dev , n , & pavgroup - > aliaslist ,
alias_list ) {
2015-08-07 13:20:16 +02:00
spin_lock_irqsave ( get_ccwdev_lock ( dev - > cdev ) , flags ) ;
tbcpm = dasd_eckd_cuir_remove_path ( dev , lpum , cuir ) ;
spin_unlock_irqrestore (
get_ccwdev_lock ( dev - > cdev ) , flags ) ;
if ( tbcpm < 0 )
goto out_err ;
paths | = tbcpm ;
2014-10-01 14:39:47 +02:00
}
}
2015-08-07 13:20:16 +02:00
/* notify user about all paths affected by CUIR action */
dasd_eckd_cuir_notify_user ( device , paths , sch_id , CUIR_QUIESCE ) ;
return 0 ;
out_err :
return tbcpm ;
2014-10-01 14:39:47 +02:00
}
static int dasd_eckd_cuir_resume ( struct dasd_device * device , __u8 lpum ,
2015-08-07 13:20:16 +02:00
struct subchannel_id sch_id ,
struct dasd_cuir_message * cuir )
2014-10-01 14:39:47 +02:00
{
struct alias_pav_group * pavgroup , * tempgroup ;
struct dasd_eckd_private * private ;
struct dasd_device * dev , * n ;
2015-08-07 13:20:16 +02:00
unsigned long paths = 0 ;
int tbcpm ;
2014-10-01 14:39:47 +02:00
private = ( struct dasd_eckd_private * ) device - > private ;
/*
* the path may have been added through a generic path event before
* only trigger path verification if the path is not already in use
*/
list_for_each_entry_safe ( dev , n ,
& private - > lcu - > active_devices ,
alias_list ) {
2015-08-07 13:20:16 +02:00
tbcpm = dasd_eckd_cuir_scope ( dev , lpum , cuir ) ;
paths | = tbcpm ;
if ( ! ( dev - > path_data . opm & tbcpm ) ) {
dev - > path_data . tbvpm | = tbcpm ;
2014-10-01 14:39:47 +02:00
dasd_schedule_device_bh ( dev ) ;
}
}
list_for_each_entry_safe ( dev , n ,
& private - > lcu - > inactive_devices ,
alias_list ) {
2015-08-07 13:20:16 +02:00
tbcpm = dasd_eckd_cuir_scope ( dev , lpum , cuir ) ;
paths | = tbcpm ;
if ( ! ( dev - > path_data . opm & tbcpm ) ) {
dev - > path_data . tbvpm | = tbcpm ;
2014-10-01 14:39:47 +02:00
dasd_schedule_device_bh ( dev ) ;
}
}
/* devices in PAV groups */
list_for_each_entry_safe ( pavgroup , tempgroup ,
& private - > lcu - > grouplist ,
group ) {
list_for_each_entry_safe ( dev , n ,
& pavgroup - > baselist ,
alias_list ) {
2015-08-07 13:20:16 +02:00
tbcpm = dasd_eckd_cuir_scope ( dev , lpum , cuir ) ;
paths | = tbcpm ;
if ( ! ( dev - > path_data . opm & tbcpm ) ) {
dev - > path_data . tbvpm | = tbcpm ;
2014-10-01 14:39:47 +02:00
dasd_schedule_device_bh ( dev ) ;
}
}
list_for_each_entry_safe ( dev , n ,
& pavgroup - > aliaslist ,
alias_list ) {
2015-08-07 13:20:16 +02:00
tbcpm = dasd_eckd_cuir_scope ( dev , lpum , cuir ) ;
paths | = tbcpm ;
if ( ! ( dev - > path_data . opm & tbcpm ) ) {
dev - > path_data . tbvpm | = tbcpm ;
2014-10-01 14:39:47 +02:00
dasd_schedule_device_bh ( dev ) ;
}
}
}
2015-08-07 13:20:16 +02:00
/* notify user about all paths affected by CUIR action */
dasd_eckd_cuir_notify_user ( device , paths , sch_id , CUIR_RESUME ) ;
return 0 ;
2014-10-01 14:39:47 +02:00
}
static void dasd_eckd_handle_cuir ( struct dasd_device * device , void * messages ,
__u8 lpum )
{
struct dasd_cuir_message * cuir = messages ;
struct channel_path_desc * desc ;
struct subchannel_id sch_id ;
int pos , response ;
2015-08-07 13:20:16 +02:00
DBF_DEV_EVENT ( DBF_WARNING , device ,
" CUIR request: %016llx %016llx %016llx %08x " ,
( ( u64 * ) cuir ) [ 0 ] , ( ( u64 * ) cuir ) [ 1 ] , ( ( u64 * ) cuir ) [ 2 ] ,
( ( u32 * ) cuir ) [ 3 ] ) ;
ccw_device_get_schid ( device - > cdev , & sch_id ) ;
2014-10-01 14:39:47 +02:00
/* get position of path in mask */
pos = 8 - ffs ( lpum ) ;
/* get channel path descriptor from this position */
desc = ccw_device_get_chp_desc ( device - > cdev , pos ) ;
if ( cuir - > code = = CUIR_QUIESCE ) {
/* quiesce */
2015-08-07 13:20:16 +02:00
if ( dasd_eckd_cuir_quiesce ( device , lpum , sch_id , cuir ) )
response = PSF_CUIR_LAST_PATH ;
else
response = PSF_CUIR_COMPLETED ;
2014-10-01 14:39:47 +02:00
} else if ( cuir - > code = = CUIR_RESUME ) {
/* resume */
2015-08-07 13:20:16 +02:00
dasd_eckd_cuir_resume ( device , lpum , sch_id , cuir ) ;
response = PSF_CUIR_COMPLETED ;
2014-10-01 14:39:47 +02:00
} else
response = PSF_CUIR_NOT_SUPPORTED ;
2015-08-07 13:20:16 +02:00
dasd_eckd_psf_cuir_response ( device , response ,
cuir - > message_id , desc , sch_id ) ;
DBF_DEV_EVENT ( DBF_WARNING , device ,
" CUIR response: %d on message ID %08x " , response ,
cuir - > message_id ) ;
2014-10-01 14:39:47 +02:00
/* free descriptor copy */
kfree ( desc ) ;
2015-08-07 13:20:16 +02:00
/* to make sure there is no attention left schedule work again */
device - > discipline - > check_attention ( device , lpum ) ;
2014-10-01 14:39:47 +02:00
}
static void dasd_eckd_check_attention_work ( struct work_struct * work )
{
struct check_attention_work_data * data ;
struct dasd_rssd_messages * messages ;
struct dasd_device * device ;
int rc ;
data = container_of ( work , struct check_attention_work_data , worker ) ;
device = data - > device ;
messages = kzalloc ( sizeof ( * messages ) , GFP_KERNEL ) ;
if ( ! messages ) {
DBF_DEV_EVENT ( DBF_WARNING , device , " %s " ,
" Could not allocate attention message buffer " ) ;
goto out ;
}
rc = dasd_eckd_read_message_buffer ( device , messages , data - > lpum ) ;
if ( rc )
goto out ;
if ( messages - > length = = ATTENTION_LENGTH_CUIR & &
messages - > format = = ATTENTION_FORMAT_CUIR )
dasd_eckd_handle_cuir ( device , messages , data - > lpum ) ;
out :
dasd_put_device ( device ) ;
kfree ( messages ) ;
kfree ( data ) ;
}
static int dasd_eckd_check_attention ( struct dasd_device * device , __u8 lpum )
{
struct check_attention_work_data * data ;
data = kzalloc ( sizeof ( * data ) , GFP_ATOMIC ) ;
if ( ! data )
return - ENOMEM ;
INIT_WORK ( & data - > worker , dasd_eckd_check_attention_work ) ;
dasd_get_device ( device ) ;
data - > device = device ;
data - > lpum = lpum ;
schedule_work ( & data - > worker ) ;
return 0 ;
}
2009-06-16 10:30:25 +02:00
static struct ccw_driver dasd_eckd_driver = {
2011-03-23 10:16:02 +01:00
. driver = {
. name = " dasd-eckd " ,
. owner = THIS_MODULE ,
} ,
2009-06-16 10:30:25 +02:00
. ids = dasd_eckd_ids ,
. probe = dasd_eckd_probe ,
. remove = dasd_generic_remove ,
. set_offline = dasd_generic_set_offline ,
. set_online = dasd_eckd_set_online ,
. notify = dasd_generic_notify ,
2011-01-05 12:48:03 +01:00
. path_event = dasd_generic_path_event ,
2012-06-19 17:30:12 +02:00
. shutdown = dasd_generic_shutdown ,
2009-06-16 10:30:25 +02:00
. freeze = dasd_generic_pm_freeze ,
. thaw = dasd_generic_restore_device ,
. restore = dasd_generic_restore_device ,
2010-05-26 23:27:09 +02:00
. uc_handler = dasd_generic_uc_handler ,
2013-01-02 15:18:18 +01:00
. int_class = IRQIO_DAS ,
2009-06-16 10:30:25 +02:00
} ;
2009-03-26 15:23:48 +01:00
2005-04-16 15:20:36 -07:00
/*
* max_blocks is dependent on the amount of storage that is available
* in the static io buffer for each device . Currently each device has
* 8192 bytes ( = 2 pages ) . For 64 bit one dasd_mchunkt_t structure has
* 24 bytes , the struct dasd_ccw_req has 136 bytes and each block can use
* up to 16 bytes ( 8 for the ccw and 8 for the idal pointer ) . In
* addition we have one define extent ccw + 16 bytes of data and one
* locate record ccw + 16 bytes of data . That makes :
* ( 8192 - 24 - 136 - 8 - 16 - 8 - 16 ) / 16 = 499 blocks at maximum .
* We want to fit two into the available memory so that we can immediately
* start the next request if one finishes off . That makes 249.5 blocks
* for one request . Give a little safety and the result is 240.
*/
static struct dasd_discipline dasd_eckd_discipline = {
. owner = THIS_MODULE ,
. name = " ECKD " ,
. ebcname = " ECKD " ,
2011-01-05 12:48:02 +01:00
. max_blocks = 190 ,
2005-04-16 15:20:36 -07:00
. check_device = dasd_eckd_check_characteristics ,
2008-01-26 14:11:23 +01:00
. uncheck_device = dasd_eckd_uncheck_device ,
2005-04-16 15:20:36 -07:00
. do_analysis = dasd_eckd_do_analysis ,
2011-01-05 12:48:03 +01:00
. verify_path = dasd_eckd_verify_path ,
2013-04-15 16:22:23 +02:00
. basic_to_ready = dasd_eckd_basic_to_ready ,
2008-01-26 14:11:23 +01:00
. online_to_ready = dasd_eckd_online_to_ready ,
2014-07-18 14:19:25 +02:00
. basic_to_known = dasd_eckd_basic_to_known ,
2005-04-16 15:20:36 -07:00
. fill_geometry = dasd_eckd_fill_geometry ,
. start_IO = dasd_start_IO ,
. term_IO = dasd_term_IO ,
2008-01-26 14:11:23 +01:00
. handle_terminated_request = dasd_eckd_handle_terminated_request ,
2005-04-16 15:20:36 -07:00
. format_device = dasd_eckd_format_device ,
. erp_action = dasd_eckd_erp_action ,
. erp_postaction = dasd_eckd_erp_postaction ,
2011-01-05 12:48:04 +01:00
. check_for_device_change = dasd_eckd_check_for_device_change ,
2008-01-26 14:11:23 +01:00
. build_cp = dasd_eckd_build_alias_cp ,
. free_cp = dasd_eckd_free_alias_cp ,
2005-04-16 15:20:36 -07:00
. dump_sense = dasd_eckd_dump_sense ,
2009-03-26 15:23:49 +01:00
. dump_sense_dbf = dasd_eckd_dump_sense_dbf ,
2005-04-16 15:20:36 -07:00
. fill_info = dasd_eckd_fill_info ,
2006-03-24 03:15:20 -08:00
. ioctl = dasd_eckd_ioctl ,
2009-06-16 10:30:25 +02:00
. freeze = dasd_eckd_pm_freeze ,
. restore = dasd_eckd_restore_device ,
2010-05-17 10:00:10 +02:00
. reload = dasd_eckd_reload_device ,
2010-05-17 10:00:11 +02:00
. get_uid = dasd_eckd_get_uid ,
2012-01-18 18:03:41 +01:00
. kick_validate = dasd_eckd_kick_validate_server ,
2014-10-01 14:39:47 +02:00
. check_attention = dasd_eckd_check_attention ,
2005-04-16 15:20:36 -07:00
} ;
static int __init
dasd_eckd_init ( void )
{
2009-06-12 10:26:38 +02:00
int ret ;
2005-04-16 15:20:36 -07:00
ASCEBC ( dasd_eckd_discipline . ebcname , 4 ) ;
2010-08-09 18:12:59 +02:00
dasd_reserve_req = kmalloc ( sizeof ( * dasd_reserve_req ) ,
GFP_KERNEL | GFP_DMA ) ;
if ( ! dasd_reserve_req )
return - ENOMEM ;
2011-01-05 12:48:03 +01:00
path_verification_worker = kmalloc ( sizeof ( * path_verification_worker ) ,
GFP_KERNEL | GFP_DMA ) ;
if ( ! path_verification_worker ) {
kfree ( dasd_reserve_req ) ;
return - ENOMEM ;
}
2013-08-16 15:57:32 +02:00
rawpadpage = ( void * ) __get_free_page ( GFP_KERNEL ) ;
if ( ! rawpadpage ) {
kfree ( path_verification_worker ) ;
kfree ( dasd_reserve_req ) ;
return - ENOMEM ;
}
2009-06-12 10:26:38 +02:00
ret = ccw_driver_register ( & dasd_eckd_driver ) ;
if ( ! ret )
wait_for_device_probe ( ) ;
2011-01-05 12:48:03 +01:00
else {
kfree ( path_verification_worker ) ;
2010-08-09 18:12:59 +02:00
kfree ( dasd_reserve_req ) ;
2013-08-16 15:57:32 +02:00
free_page ( ( unsigned long ) rawpadpage ) ;
2011-01-05 12:48:03 +01:00
}
2009-06-12 10:26:38 +02:00
return ret ;
2005-04-16 15:20:36 -07:00
}
static void __exit
dasd_eckd_cleanup ( void )
{
ccw_driver_unregister ( & dasd_eckd_driver ) ;
2011-01-05 12:48:03 +01:00
kfree ( path_verification_worker ) ;
2010-08-09 18:12:59 +02:00
kfree ( dasd_reserve_req ) ;
2013-08-16 15:57:32 +02:00
free_page ( ( unsigned long ) rawpadpage ) ;
2005-04-16 15:20:36 -07:00
}
module_init ( dasd_eckd_init ) ;
module_exit ( dasd_eckd_cleanup ) ;