ufs: read door bell register after clearing interrupt aggregation

In interrupt context, after reading and comparing the UTRLDBR to
hba->outstanding_request and before resetting the interrupt aggregation,
there might be completion of another transfer request (TR). Such TRs might
get stuck, pending, until the next interrupt is generated (if any).
Changing the sequence of resetting the interrupt aggregation first and
then reading UTRLDBR status, will assure that completed TRs won't get
stuck pending.

Signed-off-by: Dolev Raviv <draviv@codeaurora.org>
Signed-off-by: Christoph Hellwig <hch@lst.de>
This commit is contained in:
Dolev Raviv 2014-07-01 12:22:37 +03:00 committed by Christoph Hellwig
parent 4264fd613a
commit e9d501b154

View File

@ -2231,47 +2231,42 @@ static void ufshcd_transfer_req_compl(struct ufs_hba *hba)
u32 tr_doorbell; u32 tr_doorbell;
int result; int result;
int index; int index;
bool int_aggr_reset = false;
/* Resetting interrupt aggregation counters first and reading the
* DOOR_BELL afterward allows us to handle all the completed requests.
* In order to prevent other interrupts starvation the DB is read once
* after reset. The down side of this solution is the possibility of
* false interrupt if device completes another request after resetting
* aggregation and before reading the DB.
*/
ufshcd_reset_intr_aggr(hba);
tr_doorbell = ufshcd_readl(hba, REG_UTP_TRANSFER_REQ_DOOR_BELL); tr_doorbell = ufshcd_readl(hba, REG_UTP_TRANSFER_REQ_DOOR_BELL);
completed_reqs = tr_doorbell ^ hba->outstanding_reqs; completed_reqs = tr_doorbell ^ hba->outstanding_reqs;
for (index = 0; index < hba->nutrs; index++) { for_each_set_bit(index, &completed_reqs, hba->nutrs) {
if (test_bit(index, &completed_reqs)) { lrbp = &hba->lrb[index];
lrbp = &hba->lrb[index]; cmd = lrbp->cmd;
cmd = lrbp->cmd; if (cmd) {
/* result = ufshcd_transfer_rsp_status(hba, lrbp);
* Don't skip resetting interrupt aggregation counters scsi_dma_unmap(cmd);
* if a regular command is present. cmd->result = result;
*/ /* Mark completed command as NULL in LRB */
int_aggr_reset |= !lrbp->intr_cmd; lrbp->cmd = NULL;
clear_bit_unlock(index, &hba->lrb_in_use);
if (cmd) { /* Do not touch lrbp after scsi done */
result = ufshcd_transfer_rsp_status(hba, lrbp); cmd->scsi_done(cmd);
scsi_dma_unmap(cmd); } else if (lrbp->command_type == UTP_CMD_TYPE_DEV_MANAGE) {
cmd->result = result; if (hba->dev_cmd.complete)
/* Mark completed command as NULL in LRB */ complete(hba->dev_cmd.complete);
lrbp->cmd = NULL; }
clear_bit_unlock(index, &hba->lrb_in_use); }
/* Do not touch lrbp after scsi done */
cmd->scsi_done(cmd);
} else if (lrbp->command_type ==
UTP_CMD_TYPE_DEV_MANAGE) {
if (hba->dev_cmd.complete)
complete(hba->dev_cmd.complete);
}
} /* end of if */
} /* end of for */
/* clear corresponding bits of completed commands */ /* clear corresponding bits of completed commands */
hba->outstanding_reqs ^= completed_reqs; hba->outstanding_reqs ^= completed_reqs;
/* we might have free'd some tags above */ /* we might have free'd some tags above */
wake_up(&hba->dev_cmd.tag_wq); wake_up(&hba->dev_cmd.tag_wq);
/* Reset interrupt aggregation counters */
if (int_aggr_reset)
ufshcd_reset_intr_aggr(hba);
} }
/** /**
@ -2876,6 +2871,7 @@ static int ufshcd_abort(struct scsi_cmnd *cmd)
int poll_cnt; int poll_cnt;
u8 resp = 0xF; u8 resp = 0xF;
struct ufshcd_lrb *lrbp; struct ufshcd_lrb *lrbp;
u32 reg;
host = cmd->device->host; host = cmd->device->host;
hba = shost_priv(host); hba = shost_priv(host);
@ -2885,6 +2881,13 @@ static int ufshcd_abort(struct scsi_cmnd *cmd)
if (!(test_bit(tag, &hba->outstanding_reqs))) if (!(test_bit(tag, &hba->outstanding_reqs)))
goto out; goto out;
reg = ufshcd_readl(hba, REG_UTP_TRANSFER_REQ_DOOR_BELL);
if (!(reg & (1 << tag))) {
dev_err(hba->dev,
"%s: cmd was completed, but without a notifying intr, tag = %d",
__func__, tag);
}
lrbp = &hba->lrb[tag]; lrbp = &hba->lrb[tag];
for (poll_cnt = 100; poll_cnt; poll_cnt--) { for (poll_cnt = 100; poll_cnt; poll_cnt--) {
err = ufshcd_issue_tm_cmd(hba, lrbp->lun, lrbp->task_tag, err = ufshcd_issue_tm_cmd(hba, lrbp->lun, lrbp->task_tag,
@ -2893,8 +2896,6 @@ static int ufshcd_abort(struct scsi_cmnd *cmd)
/* cmd pending in the device */ /* cmd pending in the device */
break; break;
} else if (!err && resp == UPIU_TASK_MANAGEMENT_FUNC_COMPL) { } else if (!err && resp == UPIU_TASK_MANAGEMENT_FUNC_COMPL) {
u32 reg;
/* /*
* cmd not pending in the device, check if it is * cmd not pending in the device, check if it is
* in transition. * in transition.