2019-06-01 10:08:25 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2008-07-17 04:28:35 -04:00
/*
* sd_dif . c - SCSI Data Integrity Field
*
* Copyright ( C ) 2007 , 2008 Oracle Corporation
* Written by : Martin K . Petersen < martin . petersen @ oracle . com >
*/
2021-09-20 14:33:27 +02:00
# include <linux/blk-integrity.h>
2014-09-26 19:20:07 -04:00
# include <linux/t10-pi.h>
2008-07-17 04:28:35 -04:00
# include <scsi/scsi.h>
# include <scsi/scsi_cmnd.h>
# include <scsi/scsi_dbg.h>
# include <scsi/scsi_device.h>
# include <scsi/scsi_driver.h>
# include <scsi/scsi_eh.h>
# include <scsi/scsi_host.h>
# include <scsi/scsi_ioctl.h>
# include <scsi/scsicam.h>
# include "sd.h"
/*
* Configure exchange of protection information between OS and HBA .
*/
void sd_dif_config_host ( struct scsi_disk * sdkp )
{
struct scsi_device * sdp = sdkp - > device ;
struct gendisk * disk = sdkp - > disk ;
u8 type = sdkp - > protection_type ;
2015-10-21 13:19:33 -04:00
struct blk_integrity bi ;
2008-09-19 18:47:21 -04:00
int dif , dix ;
2008-07-17 04:28:35 -04:00
2008-09-19 18:47:21 -04:00
dif = scsi_host_dif_capable ( sdp - > host , type ) ;
dix = scsi_host_dix_capable ( sdp - > host , type ) ;
2008-07-17 04:28:35 -04:00
2008-09-19 18:47:21 -04:00
if ( ! dix & & scsi_host_dix_capable ( sdp - > host , 0 ) ) {
dif = 0 ; dix = 1 ;
}
2008-07-17 04:28:35 -04:00
2008-09-19 18:47:21 -04:00
if ( ! dix )
2008-07-17 04:28:35 -04:00
return ;
2015-10-21 13:19:33 -04:00
memset ( & bi , 0 , sizeof ( bi ) ) ;
2008-07-17 04:28:35 -04:00
/* Enable DMA of protection information */
2014-09-26 19:20:05 -04:00
if ( scsi_host_get_guard ( sdkp - > device - > host ) & SHOST_DIX_GUARD_IP ) {
2016-09-11 19:35:41 +02:00
if ( type = = T10_PI_TYPE3_PROTECTION )
2015-10-21 13:19:33 -04:00
bi . profile = & t10_pi_type3_ip ;
2008-07-17 04:28:35 -04:00
else
2015-10-21 13:19:33 -04:00
bi . profile = & t10_pi_type1_ip ;
2014-09-26 19:20:05 -04:00
2015-10-21 13:19:33 -04:00
bi . flags | = BLK_INTEGRITY_IP_CHECKSUM ;
2014-09-26 19:20:05 -04:00
} else
2016-09-11 19:35:41 +02:00
if ( type = = T10_PI_TYPE3_PROTECTION )
2015-10-21 13:19:33 -04:00
bi . profile = & t10_pi_type3_crc ;
2008-07-17 04:28:35 -04:00
else
2015-10-21 13:19:33 -04:00
bi . profile = & t10_pi_type1_crc ;
2008-07-17 04:28:35 -04:00
2015-10-21 13:19:33 -04:00
bi . tuple_size = sizeof ( struct t10_pi_tuple ) ;
2008-07-17 04:28:35 -04:00
2014-09-26 19:20:03 -04:00
if ( dif & & type ) {
2015-10-21 13:19:33 -04:00
bi . flags | = BLK_INTEGRITY_DEVICE_CAPABLE ;
2014-09-26 19:20:03 -04:00
2015-04-14 17:11:03 -04:00
if ( ! sdkp - > ATO )
2015-10-21 13:19:33 -04:00
goto out ;
2014-09-26 19:20:03 -04:00
2016-09-11 19:35:41 +02:00
if ( type = = T10_PI_TYPE3_PROTECTION )
2015-10-21 13:19:33 -04:00
bi . tag_size = sizeof ( u16 ) + sizeof ( u32 ) ;
2008-07-17 04:28:35 -04:00
else
2015-10-21 13:19:33 -04:00
bi . tag_size = sizeof ( u16 ) ;
2008-07-17 04:28:35 -04:00
}
2015-10-21 13:19:33 -04:00
2022-03-02 00:35:58 -05:00
sd_printk ( KERN_NOTICE , sdkp ,
" Enabling DIX %s, application tag size %u bytes \n " ,
bi . profile - > name , bi . tag_size ) ;
2015-10-21 13:19:33 -04:00
out :
blk_integrity_register ( disk , & bi ) ;
2008-07-17 04:28:35 -04:00
}