mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	scsi: core: Reallocate device's budget map on queue depth change
We currently use ->cmd_per_lun as initial queue depth for setting up the budget_map. Martin Wilck reported that it is common for the queue_depth to be subsequently updated in slave_configure() based on detected hardware characteristics. As a result, for some drivers, the static host template settings for cmd_per_lun and can_queue won't actually get used in practice. And if the default values are used to allocate the budget_map, memory may be consumed unnecessarily. Fix the issue by reallocating the budget_map after ->slave_configure() returns. At that time the device queue_depth should accurately reflect what the hardware needs. Link: https://lore.kernel.org/r/20220127153733.409132-1-ming.lei@redhat.com Cc: Bart Van Assche <bvanassche@acm.org> Reported-by: Martin Wilck <martin.wilck@suse.com> Suggested-by: Martin Wilck <martin.wilck@suse.com> Tested-by: Martin Wilck <mwilck@suse.com> Reviewed-by: Martin Wilck <mwilck@suse.com> Reviewed-by: Bart Van Assche <bvanassche@acm.org> Signed-off-by: Ming Lei <ming.lei@redhat.com> Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com>
This commit is contained in:
		
							parent
							
								
									936bd03405
								
							
						
					
					
						commit
						edb854a368
					
				| @ -214,6 +214,48 @@ static void scsi_unlock_floptical(struct scsi_device *sdev, | ||||
| 			 SCSI_TIMEOUT, 3, NULL); | ||||
| } | ||||
| 
 | ||||
| static int scsi_realloc_sdev_budget_map(struct scsi_device *sdev, | ||||
| 					unsigned int depth) | ||||
| { | ||||
| 	int new_shift = sbitmap_calculate_shift(depth); | ||||
| 	bool need_alloc = !sdev->budget_map.map; | ||||
| 	bool need_free = false; | ||||
| 	int ret; | ||||
| 	struct sbitmap sb_backup; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * realloc if new shift is calculated, which is caused by setting | ||||
| 	 * up one new default queue depth after calling ->slave_configure | ||||
| 	 */ | ||||
| 	if (!need_alloc && new_shift != sdev->budget_map.shift) | ||||
| 		need_alloc = need_free = true; | ||||
| 
 | ||||
| 	if (!need_alloc) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Request queue has to be frozen for reallocating budget map, | ||||
| 	 * and here disk isn't added yet, so freezing is pretty fast | ||||
| 	 */ | ||||
| 	if (need_free) { | ||||
| 		blk_mq_freeze_queue(sdev->request_queue); | ||||
| 		sb_backup = sdev->budget_map; | ||||
| 	} | ||||
| 	ret = sbitmap_init_node(&sdev->budget_map, | ||||
| 				scsi_device_max_queue_depth(sdev), | ||||
| 				new_shift, GFP_KERNEL, | ||||
| 				sdev->request_queue->node, false, true); | ||||
| 	if (need_free) { | ||||
| 		if (ret) | ||||
| 			sdev->budget_map = sb_backup; | ||||
| 		else | ||||
| 			sbitmap_free(&sb_backup); | ||||
| 		ret = 0; | ||||
| 		blk_mq_unfreeze_queue(sdev->request_queue); | ||||
| 	} | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * scsi_alloc_sdev - allocate and setup a scsi_Device | ||||
|  * @starget: which target to allocate a &scsi_device for | ||||
| @ -306,11 +348,7 @@ static struct scsi_device *scsi_alloc_sdev(struct scsi_target *starget, | ||||
| 	 * default device queue depth to figure out sbitmap shift | ||||
| 	 * since we use this queue depth most of times. | ||||
| 	 */ | ||||
| 	if (sbitmap_init_node(&sdev->budget_map, | ||||
| 				scsi_device_max_queue_depth(sdev), | ||||
| 				sbitmap_calculate_shift(depth), | ||||
| 				GFP_KERNEL, sdev->request_queue->node, | ||||
| 				false, true)) { | ||||
| 	if (scsi_realloc_sdev_budget_map(sdev, depth)) { | ||||
| 		put_device(&starget->dev); | ||||
| 		kfree(sdev); | ||||
| 		goto out; | ||||
| @ -1017,6 +1055,13 @@ static int scsi_add_lun(struct scsi_device *sdev, unsigned char *inq_result, | ||||
| 			} | ||||
| 			return SCSI_SCAN_NO_RESPONSE; | ||||
| 		} | ||||
| 
 | ||||
| 		/*
 | ||||
| 		 * The queue_depth is often changed in ->slave_configure. | ||||
| 		 * Set up budget map again since memory consumption of | ||||
| 		 * the map depends on actual queue depth. | ||||
| 		 */ | ||||
| 		scsi_realloc_sdev_budget_map(sdev, sdev->queue_depth); | ||||
| 	} | ||||
| 
 | ||||
| 	if (sdev->scsi_level >= SCSI_3) | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Ming Lei
						Ming Lei