diff options
author | Seungwon Jeon <tgih.jun@samsung.com> | 2012-07-31 20:30:30 -0400 |
---|---|---|
committer | Chris Ball <cjb@laptop.org> | 2012-09-04 13:58:08 -0400 |
commit | 182c90815993452f1902837cc342ac2c05ef13f5 (patch) | |
tree | a1c07245f4654ae67e5d6e4f161a23a55bb783c7 /drivers | |
parent | 077d40731edc90ee9dedf63249034c8cd5f694ce (diff) |
mmc: dw_mmc: amend using error interrupt status
RINTSTS status includes masked interrupts as well as unmasked.
data_status and cmd_status are set by value of RINTSTS in interrupt handler
and tasklet finally uses it to decide whether error is happened or not.
In addition, MINTSTS status is used for setting data_status in PIO.
Masked error interrupt will not be handled and that status can be considered
non-error case.
Signed-off-by: Seungwon Jeon <tgih.jun@samsung.com>
Reviewed By: Girish K S <girish.shivananjappa@linaro.org>
Acked-by: Jaehoon Chung <jh80.chung@samsung.com>
Acked-by: Will Newton <will.newton@imgtec.com>
Signed-off-by: Chris Ball <cjb@laptop.org>
Diffstat (limited to 'drivers')
-rw-r--r-- | drivers/mmc/host/dw_mmc.c | 11 |
1 files changed, 5 insertions, 6 deletions
diff --git a/drivers/mmc/host/dw_mmc.c b/drivers/mmc/host/dw_mmc.c index 72dc3cde646d..7baed457dc3b 100644 --- a/drivers/mmc/host/dw_mmc.c +++ b/drivers/mmc/host/dw_mmc.c | |||
@@ -1547,12 +1547,11 @@ static void dw_mci_cmd_interrupt(struct dw_mci *host, u32 status) | |||
1547 | static irqreturn_t dw_mci_interrupt(int irq, void *dev_id) | 1547 | static irqreturn_t dw_mci_interrupt(int irq, void *dev_id) |
1548 | { | 1548 | { |
1549 | struct dw_mci *host = dev_id; | 1549 | struct dw_mci *host = dev_id; |
1550 | u32 status, pending; | 1550 | u32 pending; |
1551 | unsigned int pass_count = 0; | 1551 | unsigned int pass_count = 0; |
1552 | int i; | 1552 | int i; |
1553 | 1553 | ||
1554 | do { | 1554 | do { |
1555 | status = mci_readl(host, RINTSTS); | ||
1556 | pending = mci_readl(host, MINTSTS); /* read-only mask reg */ | 1555 | pending = mci_readl(host, MINTSTS); /* read-only mask reg */ |
1557 | 1556 | ||
1558 | /* | 1557 | /* |
@@ -1570,7 +1569,7 @@ static irqreturn_t dw_mci_interrupt(int irq, void *dev_id) | |||
1570 | 1569 | ||
1571 | if (pending & DW_MCI_CMD_ERROR_FLAGS) { | 1570 | if (pending & DW_MCI_CMD_ERROR_FLAGS) { |
1572 | mci_writel(host, RINTSTS, DW_MCI_CMD_ERROR_FLAGS); | 1571 | mci_writel(host, RINTSTS, DW_MCI_CMD_ERROR_FLAGS); |
1573 | host->cmd_status = status; | 1572 | host->cmd_status = pending; |
1574 | smp_wmb(); | 1573 | smp_wmb(); |
1575 | set_bit(EVENT_CMD_COMPLETE, &host->pending_events); | 1574 | set_bit(EVENT_CMD_COMPLETE, &host->pending_events); |
1576 | } | 1575 | } |
@@ -1578,7 +1577,7 @@ static irqreturn_t dw_mci_interrupt(int irq, void *dev_id) | |||
1578 | if (pending & DW_MCI_DATA_ERROR_FLAGS) { | 1577 | if (pending & DW_MCI_DATA_ERROR_FLAGS) { |
1579 | /* if there is an error report DATA_ERROR */ | 1578 | /* if there is an error report DATA_ERROR */ |
1580 | mci_writel(host, RINTSTS, DW_MCI_DATA_ERROR_FLAGS); | 1579 | mci_writel(host, RINTSTS, DW_MCI_DATA_ERROR_FLAGS); |
1581 | host->data_status = status; | 1580 | host->data_status = pending; |
1582 | smp_wmb(); | 1581 | smp_wmb(); |
1583 | set_bit(EVENT_DATA_ERROR, &host->pending_events); | 1582 | set_bit(EVENT_DATA_ERROR, &host->pending_events); |
1584 | if (!(pending & (SDMMC_INT_DTO | SDMMC_INT_DCRC | | 1583 | if (!(pending & (SDMMC_INT_DTO | SDMMC_INT_DCRC | |
@@ -1589,7 +1588,7 @@ static irqreturn_t dw_mci_interrupt(int irq, void *dev_id) | |||
1589 | if (pending & SDMMC_INT_DATA_OVER) { | 1588 | if (pending & SDMMC_INT_DATA_OVER) { |
1590 | mci_writel(host, RINTSTS, SDMMC_INT_DATA_OVER); | 1589 | mci_writel(host, RINTSTS, SDMMC_INT_DATA_OVER); |
1591 | if (!host->data_status) | 1590 | if (!host->data_status) |
1592 | host->data_status = status; | 1591 | host->data_status = pending; |
1593 | smp_wmb(); | 1592 | smp_wmb(); |
1594 | if (host->dir_status == DW_MCI_RECV_STATUS) { | 1593 | if (host->dir_status == DW_MCI_RECV_STATUS) { |
1595 | if (host->sg != NULL) | 1594 | if (host->sg != NULL) |
@@ -1613,7 +1612,7 @@ static irqreturn_t dw_mci_interrupt(int irq, void *dev_id) | |||
1613 | 1612 | ||
1614 | if (pending & SDMMC_INT_CMD_DONE) { | 1613 | if (pending & SDMMC_INT_CMD_DONE) { |
1615 | mci_writel(host, RINTSTS, SDMMC_INT_CMD_DONE); | 1614 | mci_writel(host, RINTSTS, SDMMC_INT_CMD_DONE); |
1616 | dw_mci_cmd_interrupt(host, status); | 1615 | dw_mci_cmd_interrupt(host, pending); |
1617 | } | 1616 | } |
1618 | 1617 | ||
1619 | if (pending & SDMMC_INT_CD) { | 1618 | if (pending & SDMMC_INT_CD) { |