aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/ata
diff options
context:
space:
mode:
authorTejun Heo <htejun@gmail.com>2007-09-23 00:14:12 -0400
committerJeff Garzik <jeff@garzik.org>2007-10-12 14:55:41 -0400
commit31cc23b34913bc173680bdc87af79e551bf8cc0d (patch)
treeec64421ead9259174f0de8b22c36449ece6d69a4 /drivers/ata
parentfb7fd61454c8681cd2621051a710b78a00369203 (diff)
libata-pmp-prep: implement ops->qc_defer()
Controllers which support PMP have various restrictions on which combinations of commands are allowed to what number of devices concurrently. This patch implements ops->qc_defer() which determines whether a qc can be issued at the moment or should be deferred. If the function returns ATA_DEFER_LINK, the qc will be deferred until a qc completes on the link. If ATA_DEFER_PORT, until a qc completes on any link. The defer conditions are advisory and in general ATA_DEFER_LINK can be considered as lower priority deferring than ATA_DEFER_PORT. ops->qc_defer() replaces fixed ata_scmd_need_defer(). For standard NCQ/non-NCQ exclusion, ata_std_qc_defer() is implemented. ahci and sata_sil24 are converted to use ata_std_qc_defer(). ops->qc_defer() is heavier than the original mechanism because full qc is prepped before determining to defer it, but various information is needed to determine defer conditinos and fully translating a qc is the only way to supply such information in generic manner. IMHO, this shouldn't cause any noticeable performance issues as * for most cases deferring occurs rarely (except for NCQ-aware cmd-switching PMP) * translation itself isn't that expensive * once deferred the command won't be repeated until another command completes which usually is a very long time cpu-wise. Signed-off-by: Tejun Heo <htejun@gmail.com> Signed-off-by: Jeff Garzik <jeff@garzik.org>
Diffstat (limited to 'drivers/ata')
-rw-r--r--drivers/ata/ahci.c2
-rw-r--r--drivers/ata/libata-core.c31
-rw-r--r--drivers/ata/libata-scsi.c62
-rw-r--r--drivers/ata/sata_nv.c1
-rw-r--r--drivers/ata/sata_sil24.c1
5 files changed, 61 insertions, 36 deletions
diff --git a/drivers/ata/ahci.c b/drivers/ata/ahci.c
index 0a6b694f0d3a..cf3404467ceb 100644
--- a/drivers/ata/ahci.c
+++ b/drivers/ata/ahci.c
@@ -268,6 +268,7 @@ static const struct ata_port_operations ahci_ops = {
268 268
269 .tf_read = ahci_tf_read, 269 .tf_read = ahci_tf_read,
270 270
271 .qc_defer = ata_std_qc_defer,
271 .qc_prep = ahci_qc_prep, 272 .qc_prep = ahci_qc_prep,
272 .qc_issue = ahci_qc_issue, 273 .qc_issue = ahci_qc_issue,
273 274
@@ -298,6 +299,7 @@ static const struct ata_port_operations ahci_vt8251_ops = {
298 299
299 .tf_read = ahci_tf_read, 300 .tf_read = ahci_tf_read,
300 301
302 .qc_defer = ata_std_qc_defer,
301 .qc_prep = ahci_qc_prep, 303 .qc_prep = ahci_qc_prep,
302 .qc_issue = ahci_qc_issue, 304 .qc_issue = ahci_qc_issue,
303 305
diff --git a/drivers/ata/libata-core.c b/drivers/ata/libata-core.c
index 9467c2f60192..b666f51da7ed 100644
--- a/drivers/ata/libata-core.c
+++ b/drivers/ata/libata-core.c
@@ -4346,6 +4346,36 @@ int ata_check_atapi_dma(struct ata_queued_cmd *qc)
4346} 4346}
4347 4347
4348/** 4348/**
4349 * ata_std_qc_defer - Check whether a qc needs to be deferred
4350 * @qc: ATA command in question
4351 *
4352 * Non-NCQ commands cannot run with any other command, NCQ or
4353 * not. As upper layer only knows the queue depth, we are
4354 * responsible for maintaining exclusion. This function checks
4355 * whether a new command @qc can be issued.
4356 *
4357 * LOCKING:
4358 * spin_lock_irqsave(host lock)
4359 *
4360 * RETURNS:
4361 * ATA_DEFER_* if deferring is needed, 0 otherwise.
4362 */
4363int ata_std_qc_defer(struct ata_queued_cmd *qc)
4364{
4365 struct ata_link *link = qc->dev->link;
4366
4367 if (qc->tf.protocol == ATA_PROT_NCQ) {
4368 if (!ata_tag_valid(link->active_tag))
4369 return 0;
4370 } else {
4371 if (!ata_tag_valid(link->active_tag) && !link->sactive)
4372 return 0;
4373 }
4374
4375 return ATA_DEFER_LINK;
4376}
4377
4378/**
4349 * ata_qc_prep - Prepare taskfile for submission 4379 * ata_qc_prep - Prepare taskfile for submission
4350 * @qc: Metadata associated with taskfile to be prepared 4380 * @qc: Metadata associated with taskfile to be prepared
4351 * 4381 *
@@ -7111,6 +7141,7 @@ EXPORT_SYMBOL_GPL(ata_interrupt);
7111EXPORT_SYMBOL_GPL(ata_do_set_mode); 7141EXPORT_SYMBOL_GPL(ata_do_set_mode);
7112EXPORT_SYMBOL_GPL(ata_data_xfer); 7142EXPORT_SYMBOL_GPL(ata_data_xfer);
7113EXPORT_SYMBOL_GPL(ata_data_xfer_noirq); 7143EXPORT_SYMBOL_GPL(ata_data_xfer_noirq);
7144EXPORT_SYMBOL_GPL(ata_std_qc_defer);
7114EXPORT_SYMBOL_GPL(ata_qc_prep); 7145EXPORT_SYMBOL_GPL(ata_qc_prep);
7115EXPORT_SYMBOL_GPL(ata_dumb_qc_prep); 7146EXPORT_SYMBOL_GPL(ata_dumb_qc_prep);
7116EXPORT_SYMBOL_GPL(ata_noop_qc_prep); 7147EXPORT_SYMBOL_GPL(ata_noop_qc_prep);
diff --git a/drivers/ata/libata-scsi.c b/drivers/ata/libata-scsi.c
index dc274001ddd9..8ca2caeed017 100644
--- a/drivers/ata/libata-scsi.c
+++ b/drivers/ata/libata-scsi.c
@@ -749,6 +749,13 @@ static void ata_scsi_sdev_config(struct scsi_device *sdev)
749{ 749{
750 sdev->use_10_for_rw = 1; 750 sdev->use_10_for_rw = 1;
751 sdev->use_10_for_ms = 1; 751 sdev->use_10_for_ms = 1;
752
753 /* Schedule policy is determined by ->qc_defer() callback and
754 * it needs to see every deferred qc. Set dev_blocked to 1 to
755 * prevent SCSI midlayer from automatically deferring
756 * requests.
757 */
758 sdev->max_device_blocked = 1;
752} 759}
753 760
754static void ata_scsi_dev_config(struct scsi_device *sdev, 761static void ata_scsi_dev_config(struct scsi_device *sdev,
@@ -1416,37 +1423,6 @@ static void ata_scsi_qc_complete(struct ata_queued_cmd *qc)
1416} 1423}
1417 1424
1418/** 1425/**
1419 * ata_scmd_need_defer - Check whether we need to defer scmd
1420 * @dev: ATA device to which the command is addressed
1421 * @is_io: Is the command IO (and thus possibly NCQ)?
1422 *
1423 * NCQ and non-NCQ commands cannot run together. As upper layer
1424 * only knows the queue depth, we are responsible for maintaining
1425 * exclusion. This function checks whether a new command can be
1426 * issued to @dev.
1427 *
1428 * LOCKING:
1429 * spin_lock_irqsave(host lock)
1430 *
1431 * RETURNS:
1432 * 1 if deferring is needed, 0 otherwise.
1433 */
1434static int ata_scmd_need_defer(struct ata_device *dev, int is_io)
1435{
1436 struct ata_link *link = dev->link;
1437 int is_ncq = is_io && ata_ncq_enabled(dev);
1438
1439 if (is_ncq) {
1440 if (!ata_tag_valid(link->active_tag))
1441 return 0;
1442 } else {
1443 if (!ata_tag_valid(link->active_tag) && !link->sactive)
1444 return 0;
1445 }
1446 return 1;
1447}
1448
1449/**
1450 * ata_scsi_translate - Translate then issue SCSI command to ATA device 1426 * ata_scsi_translate - Translate then issue SCSI command to ATA device
1451 * @dev: ATA device to which the command is addressed 1427 * @dev: ATA device to which the command is addressed
1452 * @cmd: SCSI command to execute 1428 * @cmd: SCSI command to execute
@@ -1477,14 +1453,12 @@ static int ata_scsi_translate(struct ata_device *dev, struct scsi_cmnd *cmd,
1477 void (*done)(struct scsi_cmnd *), 1453 void (*done)(struct scsi_cmnd *),
1478 ata_xlat_func_t xlat_func) 1454 ata_xlat_func_t xlat_func)
1479{ 1455{
1456 struct ata_port *ap = dev->link->ap;
1480 struct ata_queued_cmd *qc; 1457 struct ata_queued_cmd *qc;
1481 int is_io = xlat_func == ata_scsi_rw_xlat; 1458 int rc;
1482 1459
1483 VPRINTK("ENTER\n"); 1460 VPRINTK("ENTER\n");
1484 1461
1485 if (unlikely(ata_scmd_need_defer(dev, is_io)))
1486 goto defer;
1487
1488 qc = ata_scsi_qc_new(dev, cmd, done); 1462 qc = ata_scsi_qc_new(dev, cmd, done);
1489 if (!qc) 1463 if (!qc)
1490 goto err_mem; 1464 goto err_mem;
@@ -1508,6 +1482,11 @@ static int ata_scsi_translate(struct ata_device *dev, struct scsi_cmnd *cmd,
1508 if (xlat_func(qc)) 1482 if (xlat_func(qc))
1509 goto early_finish; 1483 goto early_finish;
1510 1484
1485 if (ap->ops->qc_defer) {
1486 if ((rc = ap->ops->qc_defer(qc)))
1487 goto defer;
1488 }
1489
1511 /* select device, send command to hardware */ 1490 /* select device, send command to hardware */
1512 ata_qc_issue(qc); 1491 ata_qc_issue(qc);
1513 1492
@@ -1529,8 +1508,12 @@ err_mem:
1529 return 0; 1508 return 0;
1530 1509
1531defer: 1510defer:
1511 ata_qc_free(qc);
1532 DPRINTK("EXIT - defer\n"); 1512 DPRINTK("EXIT - defer\n");
1533 return SCSI_MLQUEUE_DEVICE_BUSY; 1513 if (rc == ATA_DEFER_LINK)
1514 return SCSI_MLQUEUE_DEVICE_BUSY;
1515 else
1516 return SCSI_MLQUEUE_HOST_BUSY;
1534} 1517}
1535 1518
1536/** 1519/**
@@ -3034,6 +3017,13 @@ int ata_scsi_add_hosts(struct ata_host *host, struct scsi_host_template *sht)
3034 shost->max_channel = 1; 3017 shost->max_channel = 1;
3035 shost->max_cmd_len = 16; 3018 shost->max_cmd_len = 16;
3036 3019
3020 /* Schedule policy is determined by ->qc_defer()
3021 * callback and it needs to see every deferred qc.
3022 * Set host_blocked to 1 to prevent SCSI midlayer from
3023 * automatically deferring requests.
3024 */
3025 shost->max_host_blocked = 1;
3026
3037 rc = scsi_add_host(ap->scsi_host, ap->host->dev); 3027 rc = scsi_add_host(ap->scsi_host, ap->host->dev);
3038 if (rc) 3028 if (rc)
3039 goto err_add; 3029 goto err_add;
diff --git a/drivers/ata/sata_nv.c b/drivers/ata/sata_nv.c
index b860f99fc288..40557fe2ffdf 100644
--- a/drivers/ata/sata_nv.c
+++ b/drivers/ata/sata_nv.c
@@ -423,6 +423,7 @@ static const struct ata_port_operations nv_adma_ops = {
423 .bmdma_start = ata_bmdma_start, 423 .bmdma_start = ata_bmdma_start,
424 .bmdma_stop = ata_bmdma_stop, 424 .bmdma_stop = ata_bmdma_stop,
425 .bmdma_status = ata_bmdma_status, 425 .bmdma_status = ata_bmdma_status,
426 .qc_defer = ata_std_qc_defer,
426 .qc_prep = nv_adma_qc_prep, 427 .qc_prep = nv_adma_qc_prep,
427 .qc_issue = nv_adma_qc_issue, 428 .qc_issue = nv_adma_qc_issue,
428 .freeze = nv_adma_freeze, 429 .freeze = nv_adma_freeze,
diff --git a/drivers/ata/sata_sil24.c b/drivers/ata/sata_sil24.c
index d9c010ab2280..9acfce43bde4 100644
--- a/drivers/ata/sata_sil24.c
+++ b/drivers/ata/sata_sil24.c
@@ -393,6 +393,7 @@ static const struct ata_port_operations sil24_ops = {
393 393
394 .tf_read = sil24_tf_read, 394 .tf_read = sil24_tf_read,
395 395
396 .qc_defer = ata_std_qc_defer,
396 .qc_prep = sil24_qc_prep, 397 .qc_prep = sil24_qc_prep,
397 .qc_issue = sil24_qc_issue, 398 .qc_issue = sil24_qc_issue,
398 399