From 5d1646d90e1f2cceb9f0828f4b28318cd0ec7744 Mon Sep 17 00:00:00 2001 From: Daniel Baumann Date: Sat, 27 Apr 2024 12:05:51 +0200 Subject: Adding upstream version 5.10.209. Signed-off-by: Daniel Baumann --- drivers/scsi/sd_dif.c | 83 +++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 83 insertions(+) create mode 100644 drivers/scsi/sd_dif.c (limited to 'drivers/scsi/sd_dif.c') diff --git a/drivers/scsi/sd_dif.c b/drivers/scsi/sd_dif.c new file mode 100644 index 000000000..4cadb2607 --- /dev/null +++ b/drivers/scsi/sd_dif.c @@ -0,0 +1,83 @@ +// SPDX-License-Identifier: GPL-2.0-only +/* + * sd_dif.c - SCSI Data Integrity Field + * + * Copyright (C) 2007, 2008 Oracle Corporation + * Written by: Martin K. Petersen + */ + +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#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; + struct blk_integrity bi; + int dif, dix; + + dif = scsi_host_dif_capable(sdp->host, type); + dix = scsi_host_dix_capable(sdp->host, type); + + if (!dix && scsi_host_dix_capable(sdp->host, 0)) { + dif = 0; dix = 1; + } + + if (!dix) + return; + + memset(&bi, 0, sizeof(bi)); + + /* Enable DMA of protection information */ + if (scsi_host_get_guard(sdkp->device->host) & SHOST_DIX_GUARD_IP) { + if (type == T10_PI_TYPE3_PROTECTION) + bi.profile = &t10_pi_type3_ip; + else + bi.profile = &t10_pi_type1_ip; + + bi.flags |= BLK_INTEGRITY_IP_CHECKSUM; + } else + if (type == T10_PI_TYPE3_PROTECTION) + bi.profile = &t10_pi_type3_crc; + else + bi.profile = &t10_pi_type1_crc; + + bi.tuple_size = sizeof(struct t10_pi_tuple); + sd_printk(KERN_NOTICE, sdkp, + "Enabling DIX %s protection\n", bi.profile->name); + + if (dif && type) { + bi.flags |= BLK_INTEGRITY_DEVICE_CAPABLE; + + if (!sdkp->ATO) + goto out; + + if (type == T10_PI_TYPE3_PROTECTION) + bi.tag_size = sizeof(u16) + sizeof(u32); + else + bi.tag_size = sizeof(u16); + + sd_printk(KERN_NOTICE, sdkp, "DIF application tag size %u\n", + bi.tag_size); + } + +out: + blk_integrity_register(disk, &bi); +} + -- cgit v1.2.3