From: Nilay Shroff <nilay@linux.ibm.com>
To: Damien Le Moal <dlemoal@kernel.org>,
Ming Lei <ming.lei@redhat.com>,
"Martin K . Petersen" <martin.petersen@oracle.com>,
linux-scsi@vger.kernel.org
Cc: linux-block@vger.kernel.org, Christoph Hellwig <hch@lst.de>
Subject: Re: [PATCH] scsi: avoid to send scsi command with ->queue_limits lock held
Date: Sat, 4 Jan 2025 16:58:48 +0530 [thread overview]
Message-ID: <89b19b37-3e47-4914-aed0-83e5602c3ab8@linux.ibm.com> (raw)
In-Reply-To: <770947cc-6ce9-4ef0-8577-6966c7b8d555@kernel.org>
On 1/4/25 12:47 PM, Damien Le Moal wrote:
> On 12/31/24 13:22, Ming Lei wrote:
>> Block request queue is often frozen before acquiring the queue
>> ->limits_lock.
>
> "often" is rather vague. What cases are we talking about here beside the block
> layer sysfs ->store() operations ? Fixing these is easy and does not need this
> change.
Other than sysfs ->store(), I see there're few call sites in NVMe driver (nvme_update_
ns_info_block(), nvme_update_ns_info_generic(), nvme_update_ns_info() etc.) which first
freezes queue and then acquire limits lock. Also there's one call site (__blk_mq_update_
nr_hw_queues) in block layer which does the same.
>
> Furthermore, this change almost feels like a layering violation as it replicates
> most of the queue limits structure inside sd. This introducing a strong
> dependency to the block layer internals which we should avoid.
>
In theory, we don't need to hold limits lock while sd_revalidate_disk() reads various
limits from hardware. However can't we make this one exception (till we find a better
solution) for sd_revalidate_disk() and allow it to acquire limits lock while blk-mq
request is processed?
>>
>> However, in sd_revalidate_disk(), queue_limits_start_update() is called
>> before reading all kinds of queue limits from hardware, and this way
>> causes ABBA lock easily[1][2] because queue usage counter is grabbed
>> when allocating scsi command.
>>
>> [1] https://lore.kernel.org/linux-block/Z1A8fai9_fQFhs1s@hovoldconsulting.com/
>> [2] https://lore.kernel.org/linux-scsi/ZxG38G9BuFdBpBHZ@fedora/
>>
>> Fix the issue by reading limits into one scsi disk shadow queue limits
>> structure first, then sync it to the block queue limits with
>> ->limits_lock.
>>
>> Cc: Christoph Hellwig <hch@lst.de>
>> Cc: Damien Le Moal <dlemoal@kernel.org>
>> Cc: Nilay Shroff <nilay@linux.ibm.com>
>> Fixes: 804e498e0496 ("sd: convert to the atomic queue limits API")
>> Signed-off-by: Ming Lei <ming.lei@redhat.com>
>> ---
>> drivers/scsi/sd.c | 156 +++++++++++++++++++++++++++++++-----------
>> drivers/scsi/sd.h | 59 +++++++++++++++-
>> drivers/scsi/sd_dif.c | 3 +-
>> drivers/scsi/sd_zbc.c | 14 ++--
>> 4 files changed, 181 insertions(+), 51 deletions(-)
>>
>> diff --git a/drivers/scsi/sd.c b/drivers/scsi/sd.c
>> index 8947dab132d7..6af5334dee2f 100644
>> --- a/drivers/scsi/sd.c
>> +++ b/drivers/scsi/sd.c
>> @@ -102,10 +102,10 @@ MODULE_ALIAS_SCSI_DEVICE(TYPE_ZBC);
>>
>> #define SD_MINORS 16
>>
>> -static void sd_config_discard(struct scsi_disk *sdkp, struct queue_limits *lim,
>> +static void sd_config_discard(struct scsi_disk *sdkp, struct sd_limits *lim,
>> unsigned int mode);
>> static void sd_config_write_same(struct scsi_disk *sdkp,
>> - struct queue_limits *lim);
>> + struct sd_limits *lim);
>> static int sd_revalidate_disk(struct gendisk *);
>> static void sd_unlock_native_capacity(struct gendisk *disk);
>> static void sd_shutdown(struct device *);
>> @@ -121,8 +121,64 @@ static const char *sd_cache_types[] = {
>> "write back, no read (daft)"
>> };
>>
>> +static void sd_sync_limits(struct queue_limits *blk_lim,
>> + const struct sd_limits *lim)
>> +{
>> + if (lim->has_features)
>> + blk_lim->features |= lim->features;
>> +
>> + if (lim->has_neg_features)
>> + blk_lim->features &= ~lim->neg_features;
>> +
>> + if (lim->has_alignment_offset)
>> + blk_lim->alignment_offset = lim->alignment_offset;
>> +
>> + if (lim->has_integrity)
>> + blk_lim->integrity = lim->integrity;
>> +
>> + if (lim->has_bs) {
>> + blk_lim->logical_block_size = lim->bs.logical_block_size;
>> + blk_lim->physical_block_size = lim->bs.physical_block_size;
>> + }
>> +
>> + if (lim->has_discard) {
>> + blk_lim->discard_granularity =
>> + lim->discard.discard_granularity;
>> + blk_lim->discard_alignment =
>> + lim->discard.discard_alignment;
>> + blk_lim->max_hw_discard_sectors =
>> + lim->discard.max_hw_discard_sectors;
>> + }
>> +
>> + if (lim->has_ws)
>> + blk_lim->max_write_zeroes_sectors =
>> + lim->ws.max_write_zeroes_sectors;
>> +
>> + if (lim->has_aw) {
>> + blk_lim->atomic_write_hw_max = lim->aw.atomic_write_hw_max;
>> + blk_lim->atomic_write_hw_boundary =
>> + lim->aw.atomic_write_hw_boundary;
>> + blk_lim->atomic_write_hw_unit_min =
>> + lim->aw.atomic_write_hw_unit_min;
>> + blk_lim->atomic_write_hw_unit_max =
>> + lim->aw.atomic_write_hw_unit_max;
>> + }
>> +
>> + if (lim->has_io) {
>> + blk_lim->max_dev_sectors = lim->io.max_dev_sectors;
>> + blk_lim->io_opt = lim->io.io_opt;
>> + blk_lim->io_min = lim->io.io_min;
>> + }
>> +
>> + if (lim->has_zone) {
>> + blk_lim->max_open_zones = lim->zone.max_open_zones;
>> + blk_lim->max_active_zones = lim->zone.max_active_zones;
>> + blk_lim->chunk_sectors = lim->zone.chunk_sectors;
>> + }
>> +}
>> +
>> static void sd_set_flush_flag(struct scsi_disk *sdkp,
>> - struct queue_limits *lim)
>> + struct sd_limits *lim)
>> {
>> if (sdkp->WCE) {
>> lim->features |= BLK_FEAT_WRITE_CACHE;
>> @@ -133,6 +189,7 @@ static void sd_set_flush_flag(struct scsi_disk *sdkp,
>> } else {
>> lim->features &= ~(BLK_FEAT_WRITE_CACHE | BLK_FEAT_FUA);
>> }
>> + lim->has_features = 1;
>> }
>>
>> static ssize_t
>> @@ -170,15 +227,18 @@ cache_type_store(struct device *dev, struct device_attribute *attr,
>> wce = (ct & 0x02) && !sdkp->write_prot ? 1 : 0;
>>
>> if (sdkp->cache_override) {
>> - struct queue_limits lim;
>> + struct queue_limits blk_lim;
>> + struct sd_limits lim = { 0 };
>>
>> sdkp->WCE = wce;
>> sdkp->RCD = rcd;
>>
>> - lim = queue_limits_start_update(sdkp->disk->queue);
>> sd_set_flush_flag(sdkp, &lim);
>> +
>> + blk_lim = queue_limits_start_update(sdkp->disk->queue);
>> + sd_sync_limits(&blk_lim, &lim);
>> blk_mq_freeze_queue(sdkp->disk->queue);
>> - ret = queue_limits_commit_update(sdkp->disk->queue, &lim);
>> + ret = queue_limits_commit_update(sdkp->disk->queue, &blk_lim);
>> blk_mq_unfreeze_queue(sdkp->disk->queue);
>> if (ret)
>> return ret;
>> @@ -468,7 +528,8 @@ provisioning_mode_store(struct device *dev, struct device_attribute *attr,
>> {
>> struct scsi_disk *sdkp = to_scsi_disk(dev);
>> struct scsi_device *sdp = sdkp->device;
>> - struct queue_limits lim;
>> + struct queue_limits blk_lim;
>> + struct sd_limits lim = { 0 };
>> int mode, err;
>>
>> if (!capable(CAP_SYS_ADMIN))
>> @@ -481,10 +542,11 @@ provisioning_mode_store(struct device *dev, struct device_attribute *attr,
>> if (mode < 0)
>> return -EINVAL;
>>
>> - lim = queue_limits_start_update(sdkp->disk->queue);
>> sd_config_discard(sdkp, &lim, mode);
>> + blk_lim = queue_limits_start_update(sdkp->disk->queue);
>> + sd_sync_limits(&blk_lim, &lim);
>> blk_mq_freeze_queue(sdkp->disk->queue);
>> - err = queue_limits_commit_update(sdkp->disk->queue, &lim);
>> + err = queue_limits_commit_update(sdkp->disk->queue, &blk_lim);
>> blk_mq_unfreeze_queue(sdkp->disk->queue);
>> if (err)
>> return err;
>> @@ -570,7 +632,8 @@ max_write_same_blocks_store(struct device *dev, struct device_attribute *attr,
>> {
>> struct scsi_disk *sdkp = to_scsi_disk(dev);
>> struct scsi_device *sdp = sdkp->device;
>> - struct queue_limits lim;
>> + struct queue_limits blk_lim;
>> + struct sd_limits lim = { 0 };
>> unsigned long max;
>> int err;
>>
>> @@ -592,10 +655,11 @@ max_write_same_blocks_store(struct device *dev, struct device_attribute *attr,
>> sdkp->max_ws_blocks = max;
>> }
>>
>> - lim = queue_limits_start_update(sdkp->disk->queue);
>> sd_config_write_same(sdkp, &lim);
>> + blk_lim = queue_limits_start_update(sdkp->disk->queue);
>> + sd_sync_limits(&blk_lim, &lim);
>> blk_mq_freeze_queue(sdkp->disk->queue);
>> - err = queue_limits_commit_update(sdkp->disk->queue, &lim);
>> + err = queue_limits_commit_update(sdkp->disk->queue, &blk_lim);
>> blk_mq_unfreeze_queue(sdkp->disk->queue);
>> if (err)
>> return err;
>> @@ -847,14 +911,14 @@ static void sd_disable_discard(struct scsi_disk *sdkp)
>> blk_queue_disable_discard(sdkp->disk->queue);
>> }
>>
>> -static void sd_config_discard(struct scsi_disk *sdkp, struct queue_limits *lim,
>> +static void sd_config_discard(struct scsi_disk *sdkp, struct sd_limits *lim,
>> unsigned int mode)
>> {
>> unsigned int logical_block_size = sdkp->device->sector_size;
>> unsigned int max_blocks = 0;
>>
>> - lim->discard_alignment = sdkp->unmap_alignment * logical_block_size;
>> - lim->discard_granularity = max(sdkp->physical_block_size,
>> + lim->discard.discard_alignment = sdkp->unmap_alignment * logical_block_size;
>> + lim->discard.discard_granularity = max(sdkp->physical_block_size,
>> sdkp->unmap_granularity * logical_block_size);
>> sdkp->provisioning_mode = mode;
>>
>> @@ -893,8 +957,9 @@ static void sd_config_discard(struct scsi_disk *sdkp, struct queue_limits *lim,
>> break;
>> }
>>
>> - lim->max_hw_discard_sectors = max_blocks *
>> + lim->discard.max_hw_discard_sectors = max_blocks *
>> (logical_block_size >> SECTOR_SHIFT);
>> + lim->has_discard = 1;
>> }
>>
>> static void *sd_set_special_bvec(struct request *rq, unsigned int data_len)
>> @@ -940,7 +1005,7 @@ static blk_status_t sd_setup_unmap_cmnd(struct scsi_cmnd *cmd)
>> return scsi_alloc_sgtables(cmd);
>> }
>>
>> -static void sd_config_atomic(struct scsi_disk *sdkp, struct queue_limits *lim)
>> +static void sd_config_atomic(struct scsi_disk *sdkp, struct sd_limits *lim)
>> {
>> unsigned int logical_block_size = sdkp->device->sector_size,
>> physical_block_size_sectors, max_atomic, unit_min, unit_max;
>> @@ -992,10 +1057,11 @@ static void sd_config_atomic(struct scsi_disk *sdkp, struct queue_limits *lim)
>> return;
>> }
>>
>> - lim->atomic_write_hw_max = max_atomic * logical_block_size;
>> - lim->atomic_write_hw_boundary = 0;
>> - lim->atomic_write_hw_unit_min = unit_min * logical_block_size;
>> - lim->atomic_write_hw_unit_max = unit_max * logical_block_size;
>> + lim->aw.atomic_write_hw_max = max_atomic * logical_block_size;
>> + lim->aw.atomic_write_hw_boundary = 0;
>> + lim->aw.atomic_write_hw_unit_min = unit_min * logical_block_size;
>> + lim->aw.atomic_write_hw_unit_max = unit_max * logical_block_size;
>> + lim->has_aw = 1;
>> }
>>
>> static blk_status_t sd_setup_write_same16_cmnd(struct scsi_cmnd *cmd,
>> @@ -1088,7 +1154,7 @@ static void sd_disable_write_same(struct scsi_disk *sdkp)
>> }
>>
>> static void sd_config_write_same(struct scsi_disk *sdkp,
>> - struct queue_limits *lim)
>> + struct sd_limits *lim)
>> {
>> unsigned int logical_block_size = sdkp->device->sector_size;
>>
>> @@ -1143,8 +1209,9 @@ static void sd_config_write_same(struct scsi_disk *sdkp,
>> }
>>
>> out:
>> - lim->max_write_zeroes_sectors =
>> + lim->ws.max_write_zeroes_sectors =
>> sdkp->max_ws_blocks * (logical_block_size >> SECTOR_SHIFT);
>> + lim->has_ws = 1;
>> }
>>
>> static blk_status_t sd_setup_flush_cmnd(struct scsi_cmnd *cmd)
>> @@ -2574,7 +2641,7 @@ static int sd_read_protection_type(struct scsi_disk *sdkp, unsigned char *buffer
>> }
>>
>> static void sd_config_protection(struct scsi_disk *sdkp,
>> - struct queue_limits *lim)
>> + struct sd_limits *lim)
>> {
>> struct scsi_device *sdp = sdkp->device;
>>
>> @@ -2628,7 +2695,7 @@ static void read_capacity_error(struct scsi_disk *sdkp, struct scsi_device *sdp,
>> #define READ_CAPACITY_RETRIES_ON_RESET 10
>>
>> static int read_capacity_16(struct scsi_disk *sdkp, struct scsi_device *sdp,
>> - struct queue_limits *lim, unsigned char *buffer)
>> + struct sd_limits *lim, unsigned char *buffer)
>> {
>> unsigned char cmd[16];
>> struct scsi_sense_hdr sshdr;
>> @@ -2703,6 +2770,7 @@ static int read_capacity_16(struct scsi_disk *sdkp, struct scsi_device *sdp,
>> /* Lowest aligned logical block */
>> alignment = ((buffer[14] & 0x3f) << 8 | buffer[15]) * sector_size;
>> lim->alignment_offset = alignment;
>> + lim->has_alignment_offset = 1;
>> if (alignment && sdkp->first_scan)
>> sd_printk(KERN_NOTICE, sdkp,
>> "physical block alignment offset: %u\n", alignment);
>> @@ -2814,7 +2882,7 @@ static int sd_try_rc16_first(struct scsi_device *sdp)
>> * read disk capacity
>> */
>> static void
>> -sd_read_capacity(struct scsi_disk *sdkp, struct queue_limits *lim,
>> +sd_read_capacity(struct scsi_disk *sdkp, struct sd_limits *lim,
>> unsigned char *buffer)
>> {
>> int sector_size;
>> @@ -2900,8 +2968,9 @@ sd_read_capacity(struct scsi_disk *sdkp, struct queue_limits *lim,
>> */
>> sector_size = 512;
>> }
>> - lim->logical_block_size = sector_size;
>> - lim->physical_block_size = sdkp->physical_block_size;
>> + lim->bs.logical_block_size = sector_size;
>> + lim->bs.physical_block_size = sdkp->physical_block_size;
>> + lim->has_bs = 1;
>> sdkp->device->sector_size = sector_size;
>>
>> if (sdkp->capacity > 0xffffffff)
>> @@ -3333,7 +3402,7 @@ static unsigned int sd_discard_mode(struct scsi_disk *sdkp)
>> * Query disk device for preferred I/O sizes.
>> */
>> static void sd_read_block_limits(struct scsi_disk *sdkp,
>> - struct queue_limits *lim)
>> + struct sd_limits *lim)
>> {
>> struct scsi_vpd *vpd;
>>
>> @@ -3395,7 +3464,7 @@ static void sd_read_block_limits_ext(struct scsi_disk *sdkp)
>>
>> /* Query block device characteristics */
>> static void sd_read_block_characteristics(struct scsi_disk *sdkp,
>> - struct queue_limits *lim)
>> + struct sd_limits *lim)
>> {
>> struct scsi_vpd *vpd;
>> u16 rot;
>> @@ -3412,8 +3481,10 @@ static void sd_read_block_characteristics(struct scsi_disk *sdkp,
>> sdkp->zoned = (vpd->data[8] >> 4) & 3;
>> rcu_read_unlock();
>>
>> - if (rot == 1)
>> - lim->features &= ~(BLK_FEAT_ROTATIONAL | BLK_FEAT_ADD_RANDOM);
>> + if (rot == 1) {
>> + lim->neg_features |= (BLK_FEAT_ROTATIONAL | BLK_FEAT_ADD_RANDOM);
>> + lim->has_neg_features = 1;
>> + }
>>
>> if (!sdkp->first_scan)
>> return;
>> @@ -3700,7 +3771,8 @@ static int sd_revalidate_disk(struct gendisk *disk)
>> struct scsi_disk *sdkp = scsi_disk(disk);
>> struct scsi_device *sdp = sdkp->device;
>> sector_t old_capacity = sdkp->capacity;
>> - struct queue_limits lim;
>> + struct queue_limits blk_lim;
>> + struct sd_limits lim = { 0 };
>> unsigned char *buffer;
>> unsigned int dev_max;
>> int err;
>> @@ -3724,8 +3796,6 @@ static int sd_revalidate_disk(struct gendisk *disk)
>>
>> sd_spinup_disk(sdkp);
>>
>> - lim = queue_limits_start_update(sdkp->disk->queue);
>> -
>> /*
>> * Without media there is no reason to ask; moreover, some devices
>> * react badly if we do.
>> @@ -3746,6 +3816,7 @@ static int sd_revalidate_disk(struct gendisk *disk)
>> * doesn't support it should be treated as rotational.
>> */
>> lim.features |= (BLK_FEAT_ROTATIONAL | BLK_FEAT_ADD_RANDOM);
>> + lim.has_features = 1;
>>
>> if (scsi_device_supports_vpd(sdp)) {
>> sd_read_block_provisioning(sdkp);
>> @@ -3779,23 +3850,24 @@ static int sd_revalidate_disk(struct gendisk *disk)
>>
>> /* Some devices report a maximum block count for READ/WRITE requests. */
>> dev_max = min_not_zero(dev_max, sdkp->max_xfer_blocks);
>> - lim.max_dev_sectors = logical_to_sectors(sdp, dev_max);
>> + lim.io.max_dev_sectors = logical_to_sectors(sdp, dev_max);
>>
>> if (sd_validate_min_xfer_size(sdkp))
>> - lim.io_min = logical_to_bytes(sdp, sdkp->min_xfer_blocks);
>> + lim.io.io_min = logical_to_bytes(sdp, sdkp->min_xfer_blocks);
>> else
>> - lim.io_min = 0;
>> + lim.io.io_min = 0;
>>
>> /*
>> * Limit default to SCSI host optimal sector limit if set. There may be
>> * an impact on performance for when the size of a request exceeds this
>> * host limit.
>> */
>> - lim.io_opt = sdp->host->opt_sectors << SECTOR_SHIFT;
>> + lim.io.io_opt = sdp->host->opt_sectors << SECTOR_SHIFT;
>> if (sd_validate_opt_xfer_size(sdkp, dev_max)) {
>> - lim.io_opt = min_not_zero(lim.io_opt,
>> + lim.io.io_opt = min_not_zero(lim.io.io_opt,
>> logical_to_bytes(sdp, sdkp->opt_xfer_blocks));
>> }
>> + lim.has_io = 1;
>>
>> sdkp->first_scan = 0;
>>
>> @@ -3803,8 +3875,10 @@ static int sd_revalidate_disk(struct gendisk *disk)
>> sd_config_write_same(sdkp, &lim);
>> kfree(buffer);
>>
>> + blk_lim = queue_limits_start_update(sdkp->disk->queue);
>> + sd_sync_limits(&blk_lim, &lim);
>> blk_mq_freeze_queue(sdkp->disk->queue);
>> - err = queue_limits_commit_update(sdkp->disk->queue, &lim);
>> + err = queue_limits_commit_update(sdkp->disk->queue, &blk_lim);
>> blk_mq_unfreeze_queue(sdkp->disk->queue);
>> if (err)
>> return err;
>> diff --git a/drivers/scsi/sd.h b/drivers/scsi/sd.h
>> index 36382eca941c..68c2db27cbf3 100644
>> --- a/drivers/scsi/sd.h
>> +++ b/drivers/scsi/sd.h
>> @@ -67,6 +67,59 @@ enum {
>> SD_ZERO_WS10_UNMAP, /* Use WRITE SAME(10) with UNMAP */
>> };
>>
>> +struct sd_limits {
>> + unsigned int has_features:1;
>> + unsigned int has_neg_features:1;
>> + unsigned int has_alignment_offset:1;
>> + unsigned int has_bs:1;
>> + unsigned int has_discard:1;
>> + unsigned int has_integrity:1;
>> + unsigned int has_aw:1;
>> + unsigned int has_ws:1;
>> + unsigned int has_io:1;
>> + unsigned int has_zone:1;
>> +
>> + blk_features_t features;
>> + blk_features_t neg_features;
>> + unsigned int alignment_offset;
>> + struct blk_integrity integrity;
>> +
>> + struct {
>> + unsigned int logical_block_size;
>> + unsigned int physical_block_size;
>> + } bs;
>> +
>> + struct {
>> + unsigned int discard_granularity;
>> + unsigned int discard_alignment;
>> + unsigned int max_hw_discard_sectors;
>> + } discard;
>> +
>> + struct {
>> + unsigned int max_write_zeroes_sectors;
>> + } ws;
>> +
>> + struct {
>> + unsigned int atomic_write_hw_max;
>> + unsigned int atomic_write_hw_boundary;
>> + unsigned int atomic_write_hw_unit_min;
>> + unsigned int atomic_write_hw_unit_max;
>> + } aw;
>> +
>> + struct {
>> + unsigned int max_dev_sectors;
>> + unsigned int io_opt;
>> + unsigned int io_min;
>> + } io;
>> +
>> + struct {
>> + unsigned int zone_write_granularity;
>> + unsigned int max_open_zones;
>> + unsigned int max_active_zones;
>> + unsigned int chunk_sectors;
>> + } zone;
>> +};
>> +
>> /**
>> * struct zoned_disk_info - Specific properties of a ZBC SCSI device.
>> * @nr_zones: number of zones.
>> @@ -228,11 +281,11 @@ static inline sector_t sectors_to_logical(struct scsi_device *sdev, sector_t sec
>> return sector >> (ilog2(sdev->sector_size) - 9);
>> }
>>
>> -void sd_dif_config_host(struct scsi_disk *sdkp, struct queue_limits *lim);
>> +void sd_dif_config_host(struct scsi_disk *sdkp, struct sd_limits *lim);
>>
>> #ifdef CONFIG_BLK_DEV_ZONED
>>
>> -int sd_zbc_read_zones(struct scsi_disk *sdkp, struct queue_limits *lim,
>> +int sd_zbc_read_zones(struct scsi_disk *sdkp, struct sd_limits *lim,
>> u8 buf[SD_BUF_SIZE]);
>> int sd_zbc_revalidate_zones(struct scsi_disk *sdkp);
>> blk_status_t sd_zbc_setup_zone_mgmt_cmnd(struct scsi_cmnd *cmd,
>> @@ -245,7 +298,7 @@ int sd_zbc_report_zones(struct gendisk *disk, sector_t sector,
>> #else /* CONFIG_BLK_DEV_ZONED */
>>
>> static inline int sd_zbc_read_zones(struct scsi_disk *sdkp,
>> - struct queue_limits *lim, u8 buf[SD_BUF_SIZE])
>> + struct sd_limits *lim, u8 buf[SD_BUF_SIZE])
>> {
>> return 0;
>> }
>> diff --git a/drivers/scsi/sd_dif.c b/drivers/scsi/sd_dif.c
>> index ae6ce6f5d622..081168d4aee3 100644
>> --- a/drivers/scsi/sd_dif.c
>> +++ b/drivers/scsi/sd_dif.c
>> @@ -24,13 +24,14 @@
>> /*
>> * Configure exchange of protection information between OS and HBA.
>> */
>> -void sd_dif_config_host(struct scsi_disk *sdkp, struct queue_limits *lim)
>> +void sd_dif_config_host(struct scsi_disk *sdkp, struct sd_limits *lim)
>> {
>> struct scsi_device *sdp = sdkp->device;
>> u8 type = sdkp->protection_type;
>> struct blk_integrity *bi = &lim->integrity;
>> int dif, dix;
>>
>> + lim->has_integrity = 1;
>> memset(bi, 0, sizeof(*bi));
>>
>> dif = scsi_host_dif_capable(sdp->host, type);
>> diff --git a/drivers/scsi/sd_zbc.c b/drivers/scsi/sd_zbc.c
>> index 7a447ff600d2..c8e398a08b31 100644
>> --- a/drivers/scsi/sd_zbc.c
>> +++ b/drivers/scsi/sd_zbc.c
>> @@ -588,7 +588,7 @@ int sd_zbc_revalidate_zones(struct scsi_disk *sdkp)
>> * also the zoned device information in *sdkp. Called by sd_revalidate_disk()
>> * before the gendisk capacity has been set.
>> */
>> -int sd_zbc_read_zones(struct scsi_disk *sdkp, struct queue_limits *lim,
>> +int sd_zbc_read_zones(struct scsi_disk *sdkp, struct sd_limits *lim,
>> u8 buf[SD_BUF_SIZE])
>> {
>> unsigned int nr_zones;
>> @@ -598,6 +598,7 @@ int sd_zbc_read_zones(struct scsi_disk *sdkp, struct queue_limits *lim,
>> if (sdkp->device->type != TYPE_ZBC)
>> return 0;
>>
>> + lim->has_features = 1;
>> lim->features |= BLK_FEAT_ZONED;
>>
>> /*
>> @@ -605,7 +606,7 @@ int sd_zbc_read_zones(struct scsi_disk *sdkp, struct queue_limits *lim,
>> * zones of host-managed devices must be aligned to the device physical
>> * block size.
>> */
>> - lim->zone_write_granularity = sdkp->physical_block_size;
>> + lim->zone.zone_write_granularity = sdkp->physical_block_size;
>>
>> /* READ16/WRITE16/SYNC16 is mandatory for ZBC devices */
>> sdkp->device->use_16_for_rw = 1;
>> @@ -628,11 +629,12 @@ int sd_zbc_read_zones(struct scsi_disk *sdkp, struct queue_limits *lim,
>>
>> /* The drive satisfies the kernel restrictions: set it up */
>> if (sdkp->zones_max_open == U32_MAX)
>> - lim->max_open_zones = 0;
>> + lim->zone.max_open_zones = 0;
>> else
>> - lim->max_open_zones = sdkp->zones_max_open;
>> - lim->max_active_zones = 0;
>> - lim->chunk_sectors = logical_to_sectors(sdkp->device, zone_blocks);
>> + lim->zone.max_open_zones = sdkp->zones_max_open;
>> + lim->zone.max_active_zones = 0;
>> + lim->zone.chunk_sectors = logical_to_sectors(sdkp->device, zone_blocks);
>> + lim->has_zone = 1;
>>
>> return 0;
>>
>
>
next prev parent reply other threads:[~2025-01-04 11:29 UTC|newest]
Thread overview: 9+ messages / expand[flat|nested] mbox.gz Atom feed top
2024-12-31 4:22 [PATCH] scsi: avoid to send scsi command with ->queue_limits lock held Ming Lei
2025-01-01 11:16 ` Nilay Shroff
2025-01-02 1:09 ` Ming Lei
2025-01-04 7:17 ` Damien Le Moal
2025-01-04 11:28 ` Nilay Shroff [this message]
2025-01-04 12:50 ` Damien Le Moal
2025-01-06 1:02 ` Ming Lei
2025-01-06 1:30 ` Damien Le Moal
2025-01-06 3:01 ` Ming Lei
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=89b19b37-3e47-4914-aed0-83e5602c3ab8@linux.ibm.com \
--to=nilay@linux.ibm.com \
--cc=dlemoal@kernel.org \
--cc=hch@lst.de \
--cc=linux-block@vger.kernel.org \
--cc=linux-scsi@vger.kernel.org \
--cc=martin.petersen@oracle.com \
--cc=ming.lei@redhat.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox