mirror of
https://github.com/torvalds/linux.git
synced 2025-01-01 15:51:46 +00:00
mmc: dw_mmc: improve the timeout polling code
Just use the readl_poll_timeout{_atomic} to avold open coding them. Signed-off-by: Shawn Lin <shawn.lin@rock-chips.com> Signed-off-by: Jaehoon Chung <jh80.chung@samsung.com> Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
This commit is contained in:
parent
5a7ad1146c
commit
b6d2d81c5c
@ -19,6 +19,7 @@
|
||||
#include <linux/err.h>
|
||||
#include <linux/init.h>
|
||||
#include <linux/interrupt.h>
|
||||
#include <linux/iopoll.h>
|
||||
#include <linux/ioport.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/platform_device.h>
|
||||
@ -65,6 +66,8 @@
|
||||
|
||||
struct idmac_desc_64addr {
|
||||
u32 des0; /* Control Descriptor */
|
||||
#define IDMAC_OWN_CLR64(x) \
|
||||
!((x) & cpu_to_le32(IDMAC_DES0_OWN))
|
||||
|
||||
u32 des1; /* Reserved */
|
||||
|
||||
@ -343,7 +346,7 @@ static u32 dw_mci_prep_stop_abort(struct dw_mci *host, struct mmc_command *cmd)
|
||||
|
||||
static void dw_mci_wait_while_busy(struct dw_mci *host, u32 cmd_flags)
|
||||
{
|
||||
unsigned long timeout = jiffies + msecs_to_jiffies(500);
|
||||
u32 status;
|
||||
|
||||
/*
|
||||
* Databook says that before issuing a new data transfer command
|
||||
@ -355,14 +358,11 @@ static void dw_mci_wait_while_busy(struct dw_mci *host, u32 cmd_flags)
|
||||
*/
|
||||
if ((cmd_flags & SDMMC_CMD_PRV_DAT_WAIT) &&
|
||||
!(cmd_flags & SDMMC_CMD_VOLT_SWITCH)) {
|
||||
while (mci_readl(host, STATUS) & SDMMC_STATUS_BUSY) {
|
||||
if (time_after(jiffies, timeout)) {
|
||||
/* Command will fail; we'll pass error then */
|
||||
if (readl_poll_timeout_atomic(host->regs + SDMMC_STATUS,
|
||||
status,
|
||||
!(status & SDMMC_STATUS_BUSY),
|
||||
10, 500 * USEC_PER_MSEC))
|
||||
dev_err(host->dev, "Busy; trying anyway\n");
|
||||
break;
|
||||
}
|
||||
udelay(10);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@ -555,7 +555,7 @@ static inline int dw_mci_prepare_desc64(struct dw_mci *host,
|
||||
{
|
||||
unsigned int desc_len;
|
||||
struct idmac_desc_64addr *desc_first, *desc_last, *desc;
|
||||
unsigned long timeout;
|
||||
u32 val;
|
||||
int i;
|
||||
|
||||
desc_first = desc_last = desc = host->sg_cpu;
|
||||
@ -577,12 +577,10 @@ static inline int dw_mci_prepare_desc64(struct dw_mci *host,
|
||||
* isn't still owned by IDMAC as IDMAC's write
|
||||
* ops and CPU's read ops are asynchronous.
|
||||
*/
|
||||
timeout = jiffies + msecs_to_jiffies(100);
|
||||
while (readl(&desc->des0) & IDMAC_DES0_OWN) {
|
||||
if (time_after(jiffies, timeout))
|
||||
if (readl_poll_timeout_atomic(&desc->des0, val,
|
||||
!(val & IDMAC_DES0_OWN),
|
||||
10, 100 * USEC_PER_MSEC))
|
||||
goto err_own_bit;
|
||||
udelay(10);
|
||||
}
|
||||
|
||||
/*
|
||||
* Set the OWN bit and disable interrupts
|
||||
@ -629,7 +627,7 @@ static inline int dw_mci_prepare_desc32(struct dw_mci *host,
|
||||
{
|
||||
unsigned int desc_len;
|
||||
struct idmac_desc *desc_first, *desc_last, *desc;
|
||||
unsigned long timeout;
|
||||
u32 val;
|
||||
int i;
|
||||
|
||||
desc_first = desc_last = desc = host->sg_cpu;
|
||||
@ -651,13 +649,11 @@ static inline int dw_mci_prepare_desc32(struct dw_mci *host,
|
||||
* isn't still owned by IDMAC as IDMAC's write
|
||||
* ops and CPU's read ops are asynchronous.
|
||||
*/
|
||||
timeout = jiffies + msecs_to_jiffies(100);
|
||||
while (readl(&desc->des0) &
|
||||
cpu_to_le32(IDMAC_DES0_OWN)) {
|
||||
if (time_after(jiffies, timeout))
|
||||
if (readl_poll_timeout_atomic(&desc->des0, val,
|
||||
IDMAC_OWN_CLR64(val),
|
||||
10,
|
||||
100 * USEC_PER_MSEC))
|
||||
goto err_own_bit;
|
||||
udelay(10);
|
||||
}
|
||||
|
||||
/*
|
||||
* Set the OWN bit and disable interrupts
|
||||
@ -1136,7 +1132,6 @@ static void dw_mci_submit_data(struct dw_mci *host, struct mmc_data *data)
|
||||
static void mci_send_cmd(struct dw_mci_slot *slot, u32 cmd, u32 arg)
|
||||
{
|
||||
struct dw_mci *host = slot->host;
|
||||
unsigned long timeout = jiffies + msecs_to_jiffies(500);
|
||||
unsigned int cmd_status = 0;
|
||||
|
||||
mci_writel(host, CMDARG, arg);
|
||||
@ -1144,11 +1139,9 @@ static void mci_send_cmd(struct dw_mci_slot *slot, u32 cmd, u32 arg)
|
||||
dw_mci_wait_while_busy(host, cmd);
|
||||
mci_writel(host, CMD, SDMMC_CMD_START | cmd);
|
||||
|
||||
while (time_before(jiffies, timeout)) {
|
||||
cmd_status = mci_readl(host, CMD);
|
||||
if (!(cmd_status & SDMMC_CMD_START))
|
||||
return;
|
||||
}
|
||||
if (readl_poll_timeout_atomic(host->regs + SDMMC_CMD, cmd_status,
|
||||
!(cmd_status & SDMMC_CMD_START),
|
||||
1, 500 * USEC_PER_MSEC))
|
||||
dev_err(&slot->mmc->class_dev,
|
||||
"Timeout sending command (cmd %#x arg %#x status %#x)\n",
|
||||
cmd, arg, cmd_status);
|
||||
@ -2832,7 +2825,6 @@ no_dma:
|
||||
|
||||
static bool dw_mci_ctrl_reset(struct dw_mci *host, u32 reset)
|
||||
{
|
||||
unsigned long timeout = jiffies + msecs_to_jiffies(500);
|
||||
u32 ctrl;
|
||||
|
||||
ctrl = mci_readl(host, CTRL);
|
||||
@ -2840,17 +2832,16 @@ static bool dw_mci_ctrl_reset(struct dw_mci *host, u32 reset)
|
||||
mci_writel(host, CTRL, ctrl);
|
||||
|
||||
/* wait till resets clear */
|
||||
do {
|
||||
ctrl = mci_readl(host, CTRL);
|
||||
if (!(ctrl & reset))
|
||||
return true;
|
||||
} while (time_before(jiffies, timeout));
|
||||
|
||||
if (readl_poll_timeout_atomic(host->regs + SDMMC_CTRL, ctrl,
|
||||
!(ctrl & reset),
|
||||
1, 500 * USEC_PER_MSEC)) {
|
||||
dev_err(host->dev,
|
||||
"Timeout resetting block (ctrl reset %#x)\n",
|
||||
ctrl & reset);
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool dw_mci_reset(struct dw_mci *host)
|
||||
@ -2879,17 +2870,12 @@ static bool dw_mci_reset(struct dw_mci *host)
|
||||
|
||||
/* if using dma we wait for dma_req to clear */
|
||||
if (host->use_dma) {
|
||||
unsigned long timeout = jiffies + msecs_to_jiffies(500);
|
||||
u32 status;
|
||||
|
||||
do {
|
||||
status = mci_readl(host, STATUS);
|
||||
if (!(status & SDMMC_STATUS_DMA_REQ))
|
||||
break;
|
||||
cpu_relax();
|
||||
} while (time_before(jiffies, timeout));
|
||||
|
||||
if (status & SDMMC_STATUS_DMA_REQ) {
|
||||
if (readl_poll_timeout_atomic(host->regs + SDMMC_STATUS,
|
||||
status,
|
||||
!(status & SDMMC_STATUS_DMA_REQ),
|
||||
1, 500 * USEC_PER_MSEC)) {
|
||||
dev_err(host->dev,
|
||||
"%s: Timeout waiting for dma_req to clear during reset\n",
|
||||
__func__);
|
||||
|
Loading…
Reference in New Issue
Block a user