[PATCH] libata irq-pio: eliminate unnecessary queuing in ata_pio_first_block()

- change the return value of ata_pio_complete() 0 <-> 1
   - add return value for ata_pio_first_block()
   - rename variable "qc_completed" to "has_next" in ata_pio_task()
   - use has_next to eliminate unnecessary queuing in ata_pio_first_block()

Signed-off-by: Albert Lee <albertcc@tw.ibm.com>

==========
Signed-off-by: Jeff Garzik <jgarzik@pobox.com>
This commit is contained in:
Albert Lee
2005-11-01 19:30:05 +08:00
committed by Jeff Garzik
parent e27486db89
commit fbcdd80b0d

View File

@@ -2790,7 +2790,8 @@ static unsigned long ata_pio_poll(struct ata_port *ap)
* None. (executing in kernel thread context) * None. (executing in kernel thread context)
* *
* RETURNS: * RETURNS:
* Non-zero if qc completed, zero otherwise. * Zero if qc completed.
* Non-zero if has next.
*/ */
static int ata_pio_complete (struct ata_port *ap) static int ata_pio_complete (struct ata_port *ap)
@@ -2812,14 +2813,14 @@ static int ata_pio_complete (struct ata_port *ap)
if (drv_stat & (ATA_BUSY | ATA_DRQ)) { if (drv_stat & (ATA_BUSY | ATA_DRQ)) {
ap->hsm_task_state = HSM_ST_LAST_POLL; ap->hsm_task_state = HSM_ST_LAST_POLL;
ap->pio_task_timeout = jiffies + ATA_TMOUT_PIO; ap->pio_task_timeout = jiffies + ATA_TMOUT_PIO;
return 0; return 1;
} }
} }
drv_stat = ata_wait_idle(ap); drv_stat = ata_wait_idle(ap);
if (!ata_ok(drv_stat)) { if (!ata_ok(drv_stat)) {
ap->hsm_task_state = HSM_ST_ERR; ap->hsm_task_state = HSM_ST_ERR;
return 0; return 1;
} }
qc = ata_qc_from_tag(ap, ap->active_tag); qc = ata_qc_from_tag(ap, ap->active_tag);
@@ -2831,7 +2832,7 @@ static int ata_pio_complete (struct ata_port *ap)
/* another command may start at this point */ /* another command may start at this point */
return 1; return 0;
} }
@@ -3068,27 +3069,42 @@ static void atapi_send_cdb(struct ata_port *ap, struct ata_queued_cmd *qc)
* *
* LOCKING: * LOCKING:
* Kernel thread context (may sleep) * Kernel thread context (may sleep)
*
* RETURNS:
* Zero if irq handler takes over
* Non-zero if has next (polling).
*/ */
static void ata_pio_first_block(struct ata_port *ap) static int ata_pio_first_block(struct ata_port *ap)
{ {
struct ata_queued_cmd *qc; struct ata_queued_cmd *qc;
u8 status; u8 status;
unsigned long flags; unsigned long flags;
int has_next;
qc = ata_qc_from_tag(ap, ap->active_tag); qc = ata_qc_from_tag(ap, ap->active_tag);
assert(qc != NULL); assert(qc != NULL);
assert(qc->flags & ATA_QCFLAG_ACTIVE); assert(qc->flags & ATA_QCFLAG_ACTIVE);
/* if polling, we will stay in the work queue after sending the data.
* otherwise, interrupt handler takes over after sending the data.
*/
has_next = (qc->tf.flags & ATA_TFLAG_POLLING);
/* sleep-wait for BSY to clear */ /* sleep-wait for BSY to clear */
DPRINTK("busy wait\n"); DPRINTK("busy wait\n");
if (ata_busy_sleep(ap, ATA_TMOUT_DATAOUT_QUICK, ATA_TMOUT_DATAOUT)) if (ata_busy_sleep(ap, ATA_TMOUT_DATAOUT_QUICK, ATA_TMOUT_DATAOUT)) {
ap->hsm_task_state = HSM_ST_TMOUT;
goto err_out; goto err_out;
}
/* make sure DRQ is set */ /* make sure DRQ is set */
status = ata_chk_status(ap); status = ata_chk_status(ap);
if ((status & (ATA_BUSY | ATA_DRQ)) != ATA_DRQ) if ((status & (ATA_BUSY | ATA_DRQ)) != ATA_DRQ) {
/* device status error */
ap->hsm_task_state = HSM_ST_ERR;
goto err_out; goto err_out;
}
/* Send the CDB (atapi) or the first data block (ata pio out). /* Send the CDB (atapi) or the first data block (ata pio out).
* During the state transition, interrupt handler shouldn't * During the state transition, interrupt handler shouldn't
@@ -3112,18 +3128,15 @@ static void ata_pio_first_block(struct ata_port *ap)
/* send CDB */ /* send CDB */
atapi_send_cdb(ap, qc); atapi_send_cdb(ap, qc);
spin_unlock_irqrestore(&ap->host_set->lock, flags);
/* if polling, ata_pio_task() handles the rest. /* if polling, ata_pio_task() handles the rest.
* otherwise, interrupt handler takes over from here. * otherwise, interrupt handler takes over from here.
*/ */
if (qc->tf.flags & ATA_TFLAG_POLLING) return has_next;
queue_work(ata_wq, &ap->pio_task);
spin_unlock_irqrestore(&ap->host_set->lock, flags);
return;
err_out: err_out:
ata_pio_error(ap); return 1; /* has next */
} }
/** /**
@@ -3338,23 +3351,23 @@ static void ata_pio_task(void *_data)
{ {
struct ata_port *ap = _data; struct ata_port *ap = _data;
unsigned long timeout; unsigned long timeout;
int qc_completed; int has_next;
fsm_start: fsm_start:
timeout = 0; timeout = 0;
qc_completed = 0; has_next = 1;
switch (ap->hsm_task_state) { switch (ap->hsm_task_state) {
case HSM_ST_FIRST: case HSM_ST_FIRST:
ata_pio_first_block(ap); has_next = ata_pio_first_block(ap);
return; break;
case HSM_ST: case HSM_ST:
ata_pio_block(ap); ata_pio_block(ap);
break; break;
case HSM_ST_LAST: case HSM_ST_LAST:
qc_completed = ata_pio_complete(ap); has_next = ata_pio_complete(ap);
break; break;
case HSM_ST_POLL: case HSM_ST_POLL:
@@ -3374,7 +3387,7 @@ fsm_start:
if (timeout) if (timeout)
queue_delayed_work(ata_wq, &ap->pio_task, timeout); queue_delayed_work(ata_wq, &ap->pio_task, timeout);
else if (!qc_completed) else if (has_next)
goto fsm_start; goto fsm_start;
} }