2007-02-27 22:40:55 -05:00
# ifndef _SCSI_DISK_H
# define _SCSI_DISK_H
/*
* More than enough for everybody ; ) The huge number of majors
* is a leftover from 16 bit dev_t days , we don ' t really need that
* much numberspace .
*/
# define SD_MAJORS 16
/*
* Time out in seconds for disks and Magneto - opticals ( which are slower ) .
*/
# define SD_TIMEOUT (30 * HZ)
# define SD_MOD_TIMEOUT (75 * HZ)
2013-10-04 21:42:24 +00:00
/*
* Flush timeout is a multiplier over the standard device timeout which is
* user modifiable via sysfs but initially set to SD_TIMEOUT
*/
# define SD_FLUSH_TIMEOUT_MULTIPLIER 2
2012-09-18 12:19:32 -04:00
# define SD_WRITE_SAME_TIMEOUT (120 * HZ)
2007-02-27 22:40:55 -05:00
/*
* Number of allowed retries
*/
# define SD_MAX_RETRIES 5
# define SD_PASSTHROUGH_RETRIES 1
2012-02-09 13:48:53 -05:00
# define SD_MAX_MEDIUM_TIMEOUTS 2
2007-02-27 22:40:55 -05:00
/*
* Size of the initial data buffer for mode and read capacity data
*/
# define SD_BUF_SIZE 512
2008-08-05 21:42:21 -07:00
/*
* Number of sectors at the end of the device to avoid multi - sector
* accesses to in the case of last_sector_bug
*/
# define SD_LAST_BUGGY_SECTORS 8
2009-09-20 16:49:38 -04:00
enum {
SD_EXT_CDB_SIZE = 32 , /* Extended CDB size */
SD_MEMPOOL_SIZE = 2 , /* CDB pool size */
} ;
2012-09-18 12:19:32 -04:00
enum {
2014-06-03 18:45:51 -04:00
SD_DEF_XFER_BLOCKS = 0xffff ,
SD_MAX_XFER_BLOCKS = 0xffffffff ,
2012-09-18 12:19:32 -04:00
SD_MAX_WS10_BLOCKS = 0xffff ,
SD_MAX_WS16_BLOCKS = 0x7fffff ,
} ;
2011-03-08 02:07:15 -05:00
enum {
SD_LBP_FULL = 0 , /* Full logical block provisioning */
SD_LBP_UNMAP , /* Use UNMAP command */
SD_LBP_WS16 , /* Use WRITE SAME(16) with UNMAP bit */
SD_LBP_WS10 , /* Use WRITE SAME(10) with UNMAP bit */
SD_LBP_ZERO , /* Use WRITE SAME(10) with zero payload */
SD_LBP_DISABLE , /* Discard disabled due to failed cmd */
} ;
2007-02-27 22:40:55 -05:00
struct scsi_disk {
struct scsi_driver * driver ; /* always &sd_template */
struct scsi_device * device ;
2008-02-22 00:13:36 +01:00
struct device dev ;
2007-02-27 22:40:55 -05:00
struct gendisk * disk ;
2010-07-07 16:51:29 +02:00
atomic_t openers ;
2016-03-28 21:18:56 -04:00
sector_t capacity ; /* size in logical blocks */
2014-06-03 18:45:51 -04:00
u32 max_xfer_blocks ;
2015-11-13 16:46:48 -05:00
u32 opt_xfer_blocks ;
2011-03-08 02:07:15 -05:00
u32 max_ws_blocks ;
u32 max_unmap_blocks ;
u32 unmap_granularity ;
u32 unmap_alignment ;
2007-02-27 22:40:55 -05:00
u32 index ;
2010-09-28 14:48:47 -04:00
unsigned int physical_block_size ;
2012-02-09 13:48:53 -05:00
unsigned int max_medium_access_timeouts ;
unsigned int medium_access_timed_out ;
2007-02-27 22:40:55 -05:00
u8 media_present ;
u8 write_prot ;
2008-07-17 04:28:34 -04:00
u8 protection_type ; /* Data Integrity Field */
2011-03-08 02:07:15 -05:00
u8 provisioning_mode ;
2008-07-17 04:28:34 -04:00
unsigned ATO : 1 ; /* state of disk ATO bit */
2013-04-24 14:02:53 -07:00
unsigned cache_override : 1 ; /* temp override of WCE,RCD */
2007-02-27 22:40:55 -05:00
unsigned WCE : 1 ; /* state of disk WCE bit */
unsigned RCD : 1 ; /* state of disk RCD bit, unused */
unsigned DPOFUA : 1 ; /* state of disk DPOFUA bit */
2009-03-09 11:33:31 -04:00
unsigned first_scan : 1 ;
2011-03-08 02:07:15 -05:00
unsigned lbpme : 1 ;
unsigned lbprz : 1 ;
unsigned lbpu : 1 ;
unsigned lbpws : 1 ;
unsigned lbpws10 : 1 ;
unsigned lbpvpd : 1 ;
2013-06-06 22:15:55 -04:00
unsigned ws10 : 1 ;
2012-09-18 12:19:32 -04:00
unsigned ws16 : 1 ;
2007-02-27 22:40:55 -05:00
} ;
2008-02-22 00:13:36 +01:00
# define to_scsi_disk(obj) container_of(obj,struct scsi_disk,dev)
2007-02-27 22:40:55 -05:00
2008-06-25 11:22:41 -04:00
static inline struct scsi_disk * scsi_disk ( struct gendisk * disk )
{
return container_of ( disk - > private_data , struct scsi_disk , driver ) ;
}
2007-02-27 22:40:55 -05:00
# define sd_printk(prefix, sdsk, fmt, a...) \
( sdsk ) - > disk ? \
2014-10-24 14:26:44 +02:00
sdev_prefix_printk ( prefix , ( sdsk ) - > device , \
( sdsk ) - > disk - > disk_name , fmt , # # a ) : \
sdev_printk ( prefix , ( sdsk ) - > device , fmt , # # a )
2007-02-27 22:40:55 -05:00
2014-01-03 18:19:26 -05:00
# define sd_first_printk(prefix, sdsk, fmt, a...) \
do { \
if ( ( sdkp ) - > first_scan ) \
sd_printk ( prefix , sdsk , fmt , # # a ) ; \
} while ( 0 )
2012-02-09 13:48:53 -05:00
static inline int scsi_medium_access_command ( struct scsi_cmnd * scmd )
{
switch ( scmd - > cmnd [ 0 ] ) {
case READ_6 :
case READ_10 :
case READ_12 :
case READ_16 :
case SYNCHRONIZE_CACHE :
case VERIFY :
case VERIFY_12 :
case VERIFY_16 :
case WRITE_6 :
case WRITE_10 :
case WRITE_12 :
case WRITE_16 :
case WRITE_SAME :
case WRITE_SAME_16 :
case UNMAP :
return 1 ;
case VARIABLE_LENGTH_CMD :
switch ( scmd - > cmnd [ 9 ] ) {
case READ_32 :
case VERIFY_32 :
case WRITE_32 :
case WRITE_SAME_32 :
return 1 ;
}
}
return 0 ;
}
2016-03-28 21:18:56 -04:00
static inline sector_t logical_to_sectors ( struct scsi_device * sdev , sector_t blocks )
{
return blocks < < ( ilog2 ( sdev - > sector_size ) - 9 ) ;
}
2008-07-17 04:28:34 -04:00
/*
* A DIF - capable target device can be formatted with different
* protection schemes . Currently 0 through 3 are defined :
*
* Type 0 is regular ( unprotected ) I / O
*
* Type 1 defines the contents of the guard and reference tags
*
* Type 2 defines the contents of the guard and reference tags and
* uses 32 - byte commands to seed the latter
*
* Type 3 defines the contents of the guard tag only
*/
enum sd_dif_target_protection_types {
SD_DIF_TYPE0_PROTECTION = 0x0 ,
SD_DIF_TYPE1_PROTECTION = 0x1 ,
SD_DIF_TYPE2_PROTECTION = 0x2 ,
SD_DIF_TYPE3_PROTECTION = 0x3 ,
} ;
2014-09-26 19:20:08 -04:00
/*
* Look up the DIX operation based on whether the command is read or
* write and whether dix and dif are enabled .
*/
static inline unsigned int sd_prot_op ( bool write , bool dix , bool dif )
{
/* Lookup table: bit 2 (write), bit 1 (dix), bit 0 (dif) */
const unsigned int ops [ ] = { /* wrt dix dif */
SCSI_PROT_NORMAL , /* 0 0 0 */
SCSI_PROT_READ_STRIP , /* 0 0 1 */
SCSI_PROT_READ_INSERT , /* 0 1 0 */
SCSI_PROT_READ_PASS , /* 0 1 1 */
SCSI_PROT_NORMAL , /* 1 0 0 */
SCSI_PROT_WRITE_INSERT , /* 1 0 1 */
SCSI_PROT_WRITE_STRIP , /* 1 1 0 */
SCSI_PROT_WRITE_PASS , /* 1 1 1 */
} ;
return ops [ write < < 2 | dix < < 1 | dif ] ;
}
/*
* Returns a mask of the protection flags that are valid for a given DIX
* operation .
*/
static inline unsigned int sd_prot_flag_mask ( unsigned int prot_op )
{
const unsigned int flag_mask [ ] = {
[ SCSI_PROT_NORMAL ] = 0 ,
[ SCSI_PROT_READ_STRIP ] = SCSI_PROT_TRANSFER_PI |
SCSI_PROT_GUARD_CHECK |
SCSI_PROT_REF_CHECK |
SCSI_PROT_REF_INCREMENT ,
[ SCSI_PROT_READ_INSERT ] = SCSI_PROT_REF_INCREMENT |
SCSI_PROT_IP_CHECKSUM ,
[ SCSI_PROT_READ_PASS ] = SCSI_PROT_TRANSFER_PI |
SCSI_PROT_GUARD_CHECK |
SCSI_PROT_REF_CHECK |
SCSI_PROT_REF_INCREMENT |
SCSI_PROT_IP_CHECKSUM ,
[ SCSI_PROT_WRITE_INSERT ] = SCSI_PROT_TRANSFER_PI |
SCSI_PROT_REF_INCREMENT ,
[ SCSI_PROT_WRITE_STRIP ] = SCSI_PROT_GUARD_CHECK |
SCSI_PROT_REF_CHECK |
SCSI_PROT_REF_INCREMENT |
SCSI_PROT_IP_CHECKSUM ,
[ SCSI_PROT_WRITE_PASS ] = SCSI_PROT_TRANSFER_PI |
SCSI_PROT_GUARD_CHECK |
SCSI_PROT_REF_CHECK |
SCSI_PROT_REF_INCREMENT |
SCSI_PROT_IP_CHECKSUM ,
} ;
return flag_mask [ prot_op ] ;
}
2008-07-17 04:28:35 -04:00
/*
* Data Integrity Field tuple .
*/
struct sd_dif_tuple {
__be16 guard_tag ; /* Checksum */
__be16 app_tag ; /* Opaque storage */
__be32 ref_tag ; /* Target LBA or indirect LBA */
} ;
2008-10-13 15:20:06 -04:00
# ifdef CONFIG_BLK_DEV_INTEGRITY
2008-07-17 04:28:35 -04:00
extern void sd_dif_config_host ( struct scsi_disk * ) ;
2014-09-26 19:20:08 -04:00
extern void sd_dif_prepare ( struct scsi_cmnd * scmd ) ;
2008-07-17 04:28:35 -04:00
extern void sd_dif_complete ( struct scsi_cmnd * , unsigned int ) ;
# else /* CONFIG_BLK_DEV_INTEGRITY */
2008-10-13 15:20:06 -04:00
static inline void sd_dif_config_host ( struct scsi_disk * disk )
{
}
2014-09-26 19:20:08 -04:00
static inline int sd_dif_prepare ( struct scsi_cmnd * scmd )
2008-10-13 15:20:06 -04:00
{
return 0 ;
}
static inline void sd_dif_complete ( struct scsi_cmnd * cmd , unsigned int a )
{
}
2008-07-17 04:28:35 -04:00
# endif /* CONFIG_BLK_DEV_INTEGRITY */
2007-02-27 22:40:55 -05:00
# endif /* _SCSI_DISK_H */