Skip to content

Commit 26e85fc

Browse files
martinkpetersenJames Bottomley
authored andcommitted
[SCSI] sd: Permit merged discard requests
Support requests with more than one bio payload for discards. The total number of bytes to be discarded is stored in req->__data_len and used in sd_done() to complete the I/O. Signed-off-by: Martin K. Petersen <[email protected]> Reviewed-by: Mike Snitzer <[email protected]> Signed-off-by: James Bottomley <[email protected]>
1 parent 3c6bdae commit 26e85fc

File tree

1 file changed

+19
-13
lines changed

1 file changed

+19
-13
lines changed

drivers/scsi/sd.c

Lines changed: 19 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -583,29 +583,26 @@ static void sd_config_discard(struct scsi_disk *sdkp, unsigned int mode)
583583
}
584584

585585
/**
586-
* scsi_setup_discard_cmnd - unmap blocks on thinly provisioned device
586+
* sd_setup_discard_cmnd - unmap blocks on thinly provisioned device
587587
* @sdp: scsi device to operate one
588588
* @rq: Request to prepare
589589
*
590590
* Will issue either UNMAP or WRITE SAME(16) depending on preference
591591
* indicated by target device.
592592
**/
593-
static int scsi_setup_discard_cmnd(struct scsi_device *sdp, struct request *rq)
593+
static int sd_setup_discard_cmnd(struct scsi_device *sdp, struct request *rq)
594594
{
595595
struct scsi_disk *sdkp = scsi_disk(rq->rq_disk);
596-
struct bio *bio = rq->bio;
597-
sector_t sector = bio->bi_sector;
598-
unsigned int nr_sectors = bio_sectors(bio);
596+
sector_t sector = blk_rq_pos(rq);
597+
unsigned int nr_sectors = blk_rq_sectors(rq);
598+
unsigned int nr_bytes = blk_rq_bytes(rq);
599599
unsigned int len;
600600
int ret;
601601
char *buf;
602602
struct page *page;
603603

604-
if (sdkp->device->sector_size == 4096) {
605-
sector >>= 3;
606-
nr_sectors >>= 3;
607-
}
608-
604+
sector >>= ilog2(sdp->sector_size) - 9;
605+
nr_sectors >>= ilog2(sdp->sector_size) - 9;
609606
rq->timeout = SD_TIMEOUT;
610607

611608
memset(rq->cmd, 0, rq->cmd_len);
@@ -660,6 +657,7 @@ static int scsi_setup_discard_cmnd(struct scsi_device *sdp, struct request *rq)
660657
blk_add_request_payload(rq, page, len);
661658
ret = scsi_setup_blk_pc_cmnd(sdp, rq);
662659
rq->buffer = page_address(page);
660+
rq->__data_len = nr_bytes;
663661

664662
out:
665663
if (ret != BLKPREP_OK) {
@@ -712,7 +710,7 @@ static int sd_prep_fn(struct request_queue *q, struct request *rq)
712710
* block PC requests to make life easier.
713711
*/
714712
if (rq->cmd_flags & REQ_DISCARD) {
715-
ret = scsi_setup_discard_cmnd(sdp, rq);
713+
ret = sd_setup_discard_cmnd(sdp, rq);
716714
goto out;
717715
} else if (rq->cmd_flags & REQ_FLUSH) {
718716
ret = scsi_setup_flush_cmnd(sdp, rq);
@@ -1482,12 +1480,20 @@ static int sd_done(struct scsi_cmnd *SCpnt)
14821480
unsigned int good_bytes = result ? 0 : scsi_bufflen(SCpnt);
14831481
struct scsi_sense_hdr sshdr;
14841482
struct scsi_disk *sdkp = scsi_disk(SCpnt->request->rq_disk);
1483+
struct request *req = SCpnt->request;
14851484
int sense_valid = 0;
14861485
int sense_deferred = 0;
14871486
unsigned char op = SCpnt->cmnd[0];
14881487

1489-
if ((SCpnt->request->cmd_flags & REQ_DISCARD) && !result)
1490-
scsi_set_resid(SCpnt, 0);
1488+
if (req->cmd_flags & REQ_DISCARD) {
1489+
if (!result) {
1490+
good_bytes = blk_rq_bytes(req);
1491+
scsi_set_resid(SCpnt, 0);
1492+
} else {
1493+
good_bytes = 0;
1494+
scsi_set_resid(SCpnt, blk_rq_bytes(req));
1495+
}
1496+
}
14911497

14921498
if (result) {
14931499
sense_valid = scsi_command_normalize_sense(SCpnt, &sshdr);

0 commit comments

Comments
 (0)