aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMartin K. Petersen <martin.petersen@oracle.com>2014-11-07 00:08:13 -0500
committerChristoph Hellwig <hch@lst.de>2014-11-12 05:19:14 -0500
commit7985090aa0201fa7760583f9f8e6ba41a8d4c392 (patch)
treebb1c70e1cd918f56068482fcc8fca298e1c7cbda
parente9afccc5245a35468f52bc3f53ed162caabf275d (diff)
sd: disable discard_zeroes_data for UNMAP
The T10 SBC UNMAP command does not provide any hard guarantees that blocks will return zeroes on a subsequent READ. This is due to the fact that the device server is free to silently ignore all or parts of the request. The only way to ensure that a block consistently returns zeroes after being unmapped is to use WRITE SAME with the UNMAP bit set. Should the device be unable to unmap one or more blocks described by the command it is required to manually write zeroes to them. Until now we have preferred UNMAP over the WRITE SAME variants to accommodate thinly provisioned devices that predated the final SBC-3 spec. This patch changes the heuristic so that we favor WRITE SAME(16) or (10) over UNMAP if these commands are marked as supported in the Logical Block Provisioning VPD page. The patch also disables discard_zeroes_data for devices operating in UNMAP mode. Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com> Reviewed-by: Paolo Bonzini <pbonzini@redhat.com> Signed-off-by: Christoph Hellwig <hch@lst.de>
-rw-r--r--drivers/scsi/sd.c10
1 files changed, 6 insertions, 4 deletions
diff --git a/drivers/scsi/sd.c b/drivers/scsi/sd.c
index b041eca8955d..95bfb7bfbb9d 100644
--- a/drivers/scsi/sd.c
+++ b/drivers/scsi/sd.c
@@ -656,7 +656,7 @@ static void sd_config_discard(struct scsi_disk *sdkp, unsigned int mode)
656 unsigned int logical_block_size = sdkp->device->sector_size; 656 unsigned int logical_block_size = sdkp->device->sector_size;
657 unsigned int max_blocks = 0; 657 unsigned int max_blocks = 0;
658 658
659 q->limits.discard_zeroes_data = sdkp->lbprz; 659 q->limits.discard_zeroes_data = 0;
660 q->limits.discard_alignment = sdkp->unmap_alignment * 660 q->limits.discard_alignment = sdkp->unmap_alignment *
661 logical_block_size; 661 logical_block_size;
662 q->limits.discard_granularity = 662 q->limits.discard_granularity =
@@ -680,11 +680,13 @@ static void sd_config_discard(struct scsi_disk *sdkp, unsigned int mode)
680 case SD_LBP_WS16: 680 case SD_LBP_WS16:
681 max_blocks = min_not_zero(sdkp->max_ws_blocks, 681 max_blocks = min_not_zero(sdkp->max_ws_blocks,
682 (u32)SD_MAX_WS16_BLOCKS); 682 (u32)SD_MAX_WS16_BLOCKS);
683 q->limits.discard_zeroes_data = sdkp->lbprz;
683 break; 684 break;
684 685
685 case SD_LBP_WS10: 686 case SD_LBP_WS10:
686 max_blocks = min_not_zero(sdkp->max_ws_blocks, 687 max_blocks = min_not_zero(sdkp->max_ws_blocks,
687 (u32)SD_MAX_WS10_BLOCKS); 688 (u32)SD_MAX_WS10_BLOCKS);
689 q->limits.discard_zeroes_data = sdkp->lbprz;
688 break; 690 break;
689 691
690 case SD_LBP_ZERO: 692 case SD_LBP_ZERO:
@@ -2622,12 +2624,12 @@ static void sd_read_block_limits(struct scsi_disk *sdkp)
2622 2624
2623 } else { /* LBP VPD page tells us what to use */ 2625 } else { /* LBP VPD page tells us what to use */
2624 2626
2625 if (sdkp->lbpu && sdkp->max_unmap_blocks) 2627 if (sdkp->lbpws)
2626 sd_config_discard(sdkp, SD_LBP_UNMAP);
2627 else if (sdkp->lbpws)
2628 sd_config_discard(sdkp, SD_LBP_WS16); 2628 sd_config_discard(sdkp, SD_LBP_WS16);
2629 else if (sdkp->lbpws10) 2629 else if (sdkp->lbpws10)
2630 sd_config_discard(sdkp, SD_LBP_WS10); 2630 sd_config_discard(sdkp, SD_LBP_WS10);
2631 else if (sdkp->lbpu && sdkp->max_unmap_blocks)
2632 sd_config_discard(sdkp, SD_LBP_UNMAP);
2631 else 2633 else
2632 sd_config_discard(sdkp, SD_LBP_DISABLE); 2634 sd_config_discard(sdkp, SD_LBP_DISABLE);
2633 } 2635 }