Commit 3ae4f265 authored by Ulf Hansson's avatar Ulf Hansson

mmc: core: Convert to use __mmc_poll_for_busy() SD_APP_OP_COND too

Similar to what has already been changed for eMMC and the MMC_SEND_OP_COND
(CMD1), let's convert the SD_APP_OP_COND (ACMD41) for SD cards to use the
common __mmc_poll_for_busy() too.

This change means the initial delay period, that starts as 10ms will now
increase for every loop when being busy. The total accepted timeout for
being busy is 1s, which is according to the SD spec.
Signed-off-by: default avatarUlf Hansson <ulf.hansson@linaro.org>
Acked-by: default avatarFelix Qin <xiaokeqinhealth@126.com>
Reviewed-by: default avatarDragan Simic <dsimic@manjaro.org>
Link: https://lore.kernel.org/r/20240425133034.79599-1-ulf.hansson@linaro.org
parent e38063b9
...@@ -19,6 +19,15 @@ ...@@ -19,6 +19,15 @@
#include "sd_ops.h" #include "sd_ops.h"
#include "mmc_ops.h" #include "mmc_ops.h"
#define SD_APP_OP_COND_PERIOD_US (10 * 1000) /* 10ms */
#define SD_APP_OP_COND_TIMEOUT_MS 1000 /* 1s */
struct sd_app_op_cond_busy_data {
struct mmc_host *host;
u32 ocr;
struct mmc_command *cmd;
};
int mmc_app_cmd(struct mmc_host *host, struct mmc_card *card) int mmc_app_cmd(struct mmc_host *host, struct mmc_card *card)
{ {
int err; int err;
...@@ -115,48 +124,63 @@ int mmc_app_set_bus_width(struct mmc_card *card, int width) ...@@ -115,48 +124,63 @@ int mmc_app_set_bus_width(struct mmc_card *card, int width)
return mmc_wait_for_app_cmd(card->host, card, &cmd); return mmc_wait_for_app_cmd(card->host, card, &cmd);
} }
int mmc_send_app_op_cond(struct mmc_host *host, u32 ocr, u32 *rocr) static int sd_app_op_cond_cb(void *cb_data, bool *busy)
{ {
struct mmc_command cmd = {}; struct sd_app_op_cond_busy_data *data = cb_data;
int i, err = 0; struct mmc_host *host = data->host;
struct mmc_command *cmd = data->cmd;
u32 ocr = data->ocr;
int err;
cmd.opcode = SD_APP_OP_COND; *busy = false;
if (mmc_host_is_spi(host))
cmd.arg = ocr & (1 << 30); /* SPI only defines one bit */
else
cmd.arg = ocr;
cmd.flags = MMC_RSP_SPI_R1 | MMC_RSP_R3 | MMC_CMD_BCR;
for (i = 100; i; i--) { err = mmc_wait_for_app_cmd(host, NULL, cmd);
err = mmc_wait_for_app_cmd(host, NULL, &cmd);
if (err) if (err)
break; return err;
/* if we're just probing, do a single pass */ /* If we're just probing, do a single pass. */
if (ocr == 0) if (ocr == 0)
break; return 0;
/* otherwise wait until reset completes */ /* Wait until reset completes. */
if (mmc_host_is_spi(host)) { if (mmc_host_is_spi(host)) {
if (!(cmd.resp[0] & R1_SPI_IDLE)) if (!(cmd->resp[0] & R1_SPI_IDLE))
break; return 0;
} else { } else if (cmd->resp[0] & MMC_CARD_BUSY) {
if (cmd.resp[0] & MMC_CARD_BUSY) return 0;
break;
} }
err = -ETIMEDOUT; *busy = true;
return 0;
}
mmc_delay(10); int mmc_send_app_op_cond(struct mmc_host *host, u32 ocr, u32 *rocr)
} {
struct mmc_command cmd = {};
struct sd_app_op_cond_busy_data cb_data = {
.host = host,
.ocr = ocr,
.cmd = &cmd
};
int err;
if (!i) cmd.opcode = SD_APP_OP_COND;
pr_err("%s: card never left busy state\n", mmc_hostname(host)); if (mmc_host_is_spi(host))
cmd.arg = ocr & (1 << 30); /* SPI only defines one bit */
else
cmd.arg = ocr;
cmd.flags = MMC_RSP_SPI_R1 | MMC_RSP_R3 | MMC_CMD_BCR;
err = __mmc_poll_for_busy(host, SD_APP_OP_COND_PERIOD_US,
SD_APP_OP_COND_TIMEOUT_MS, &sd_app_op_cond_cb,
&cb_data);
if (err)
return err;
if (rocr && !mmc_host_is_spi(host)) if (rocr && !mmc_host_is_spi(host))
*rocr = cmd.resp[0]; *rocr = cmd.resp[0];
return err; return 0;
} }
static int __mmc_send_if_cond(struct mmc_host *host, u32 ocr, u8 pcie_bits, static int __mmc_send_if_cond(struct mmc_host *host, u32 ocr, u8 pcie_bits,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment