diff options
Diffstat (limited to 'drivers/mmc')
-rw-r--r-- | drivers/mmc/host/omap_hsmmc.c | 39 |
1 files changed, 34 insertions, 5 deletions
diff --git a/drivers/mmc/host/omap_hsmmc.c b/drivers/mmc/host/omap_hsmmc.c index 6f5d3cc..f6e47a0 100644 --- a/drivers/mmc/host/omap_hsmmc.c +++ b/drivers/mmc/host/omap_hsmmc.c @@ -258,6 +258,9 @@ static void omap_hsmmc_status_notify_cb(int card_present, void *dev_id) } } +static int +omap_hsmmc_prepare_data(struct omap_hsmmc_host *host, struct mmc_request *req); + static int omap_hsmmc_card_detect(struct device *dev, int slot) { struct omap_mmc_platform_data *mmc = dev->platform_data; @@ -877,7 +880,7 @@ static DEVICE_ATTR(slot_name, S_IRUGO, omap_hsmmc_show_slot_name, NULL); */ static void omap_hsmmc_start_command(struct omap_hsmmc_host *host, struct mmc_command *cmd, - struct mmc_data *data) + struct mmc_data *data, bool no_autocmd12) { int cmdreg = 0, resptype = 0, cmdtype = 0; @@ -907,7 +910,8 @@ omap_hsmmc_start_command(struct omap_hsmmc_host *host, struct mmc_command *cmd, cmdtype = 0x3; cmdreg = (cmd->opcode << 24) | (resptype << 16) | (cmdtype << 22); - if ((host->flags & AUTO_CMD12) && mmc_op_multi(cmd->opcode)) + if ((host->flags & AUTO_CMD12) && mmc_op_multi(cmd->opcode) && + !no_autocmd12) cmdreg |= ACEN_ACMD12; if (data) { @@ -985,7 +989,7 @@ omap_hsmmc_xfer_done(struct omap_hsmmc_host *host, struct mmc_data *data) data->bytes_xfered = 0; if (data->stop && ((!(host->flags & AUTO_CMD12)) || data->error)) - omap_hsmmc_start_command(host, data->stop, NULL); + omap_hsmmc_start_command(host, data->stop, NULL, 0); else { if (data->stop) data->stop->resp[0] = OMAP_HSMMC_READ(host->base, @@ -1002,6 +1006,26 @@ omap_hsmmc_xfer_done(struct omap_hsmmc_host *host, struct mmc_data *data) static void omap_hsmmc_cmd_done(struct omap_hsmmc_host *host, struct mmc_command *cmd) { + int err; + struct mmc_request *req; + req = host->mrq; + + if ((host->mrq->sbc) && (host->cmd == host->mrq->sbc)) { + host->cmd = NULL; + err = omap_hsmmc_prepare_data(host, host->mrq); + if (err) { + req->cmd->error = err; + if (req->data) + req->data->error = err; + host->mrq = NULL; + mmc_request_done(host->mmc, req); + return; + } + omap_hsmmc_start_command(host, host->mrq->cmd, + host->mrq->data, 1); + return; + } + host->cmd = NULL; if (cmd->flags & MMC_RSP_PRESENT) { @@ -1667,6 +1691,11 @@ static void omap_hsmmc_request(struct mmc_host *mmc, struct mmc_request *req) host->reqs_blocked = 0; WARN_ON(host->mrq != NULL); host->mrq = req; + if (req->sbc) { + omap_hsmmc_start_command(host, req->sbc, NULL, 0); + return; + } + err = omap_hsmmc_prepare_data(host, req); if (err) { req->cmd->error = err; @@ -1677,7 +1706,7 @@ static void omap_hsmmc_request(struct mmc_host *mmc, struct mmc_request *req) return; } - omap_hsmmc_start_command(host, req->cmd, req->data); + omap_hsmmc_start_command(host, req->cmd, req->data, 0); } /* Routine to configure clock values. Exposed API to core */ @@ -2330,7 +2359,7 @@ static int __init omap_hsmmc_probe(struct platform_device *pdev) mmc->max_seg_size = mmc->max_req_size; mmc->caps |= MMC_CAP_MMC_HIGHSPEED | MMC_CAP_SD_HIGHSPEED | - MMC_CAP_WAIT_WHILE_BUSY | MMC_CAP_ERASE; + MMC_CAP_WAIT_WHILE_BUSY | MMC_CAP_ERASE | MMC_CAP_CMD23; mmc->caps |= mmc_slot(host).caps; if (mmc->caps & MMC_CAP_8_BIT_DATA) |