Re: [PATCH] scsi: avoid to send scsi command with ->queue_limits lock held

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



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.

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.

> 
> 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@xxxxxxxxxxxxxxxxxxxx/
> [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@xxxxxx>
> Cc: Damien Le Moal <dlemoal@xxxxxxxxxx>
> Cc: Nilay Shroff <nilay@xxxxxxxxxxxxx>
> Fixes: 804e498e0496 ("sd: convert to the atomic queue limits API")
> Signed-off-by: Ming Lei <ming.lei@xxxxxxxxxx>
> ---
>  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;
>  


-- 
Damien Le Moal
Western Digital Research




[Index of Archives]     [Linux RAID]     [Linux SCSI]     [Linux ATA RAID]     [IDE]     [Linux Wireless]     [Linux Kernel]     [ATH6KL]     [Linux Bluetooth]     [Linux Netdev]     [Kernel Newbies]     [Security]     [Git]     [Netfilter]     [Bugtraq]     [Yosemite News]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Device Mapper]

  Powered by Linux