mmc: core: Prevent violation of specs while initializing cards
authorUlf Hansson <ulf.hansson@linaro.org>
Mon, 16 Sep 2013 09:28:42 +0000 (11:28 +0200)
committerlintao <lintao@rock-chips.com>
Fri, 7 Mar 2014 09:35:47 +0000 (17:35 +0800)
According to eMMC/SD/SDIO specs, the VDD (VCC) voltage level must be
maintained during the initialization sequence. If we want/need to tune
the voltage level, a complete power cycle of the card must be executed.

Most host drivers conforms to the specifications by only allowing to
change VDD voltage level at the MMC_POWER_UP state, but some also cares
about MMC_POWER_ON state, which they should'nt. This patch will not
break those drivers, but they could clean up code to better reflect
what is expected from the protocol layer.

A big re-work of the mmc_select_voltage function is done to only change
VDD voltage level if the host supports MMC_CAP2_FULL_PWR_CYCLE.
Otherwise only validation of the host and card ocr mask will be done.

A very nice side-effect of this patch is that we now don't need to
reset the negotiated ocr mask at the mmc_power_off function, since now
it will actually reflect the present voltage level, which safely can be
used at the next power up and re-initialization. Moreover, we then only
need to execute mmc_select_voltage from the attach sequence.

Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
Signed-off-by: Chris Ball <cjb@laptop.org>
drivers/mmc/core/core.c
drivers/mmc/core/mmc.c
drivers/mmc/core/sd.c
drivers/mmc/core/sdio.c

index 3af994d4059546b5f0afe270cca307a3bccef144..2fea94cef71f2fba882e766231f6a9e99c2039c1 100644 (file)
@@ -1359,21 +1359,20 @@ u32 mmc_select_voltage(struct mmc_host *host, u32 ocr)
        int bit;
 
        ocr &= host->ocr_avail;
+       if (!ocr) {
+               dev_warn(mmc_dev(host), "no support for card's volts\n");
+               return 0;
+       }
 
-       bit = ffs(ocr);
-       if (bit) {
-               bit -= 1;
-
+       if (host->caps2 & MMC_CAP2_FULL_PWR_CYCLE) {
+               bit = ffs(ocr) - 1;
                ocr &= 3 << bit;
-
-               mmc_host_clk_hold(host);
-               host->ios.vdd = bit;
-               mmc_set_ios(host);
-               mmc_host_clk_release(host);
+               mmc_power_cycle(host, ocr);
        } else {
-               pr_warning("%s: host doesn't support card's voltages\n",
-                               mmc_hostname(host));
-               ocr = 0;
+               bit = fls(ocr) - 1;
+               ocr &= 3 << bit;
+               if (bit != host->ios.vdd)
+                       dev_warn(mmc_dev(host), "exceeding card's volts\n");
        }
 
        return ocr;
@@ -1572,14 +1571,6 @@ void mmc_power_off(struct mmc_host *host)
        host->ios.clock = 0;
        host->ios.vdd = 0;
 
-
-       /*
-        * Reset ocr mask to be the highest possible voltage supported for
-        * this card. This value will be used at next power up.
-        */
-       if (host->card)
-               host->card->ocr = 1 << (fls(host->ocr_avail) - 1);
-
        if (!mmc_host_is_spi(host)) {
                host->ios.bus_mode = MMC_BUSMODE_OPENDRAIN;
                host->ios.chip_select = MMC_CS_DONTCARE;
index eb7161c1158739cee935c8ea404f00bcc70a8f93..18ce81a85c9b8422a150a2bfd9c34c29c0acb9d4 100644 (file)
@@ -1535,7 +1535,6 @@ static int mmc_resume(struct mmc_host *host)
 
        mmc_claim_host(host);
        mmc_power_up(host, host->card->ocr);
-       mmc_select_voltage(host, host->card->ocr);
        err = mmc_init_card(host, host->card->ocr, host->card);
        mmc_release_host(host);
 
index a010ec2803e178eb80b684dd24dc0b1b2a472b08..a83debc735342cbc0319d92b381820c3066b2043 100644 (file)
@@ -1144,7 +1144,6 @@ static int mmc_sd_resume(struct mmc_host *host)
 
        mmc_claim_host(host);
        mmc_power_up(host, host->card->ocr);
-       mmc_select_voltage(host, host->card->ocr);
        err = mmc_sd_init_card(host, host->card->ocr, host->card);
        mmc_release_host(host);
 
index d5f59e15cff6c3e8ffc704cb6a6cb6684eacb527..2b5d71a24fcc68e7582fdb3799b96ec5a0e89a36 100644 (file)
@@ -1008,7 +1008,6 @@ static int mmc_sdio_resume(struct mmc_host *host)
        /* Restore power if needed */
        if (!mmc_card_keep_power(host)) {
                mmc_power_up(host, host->card->ocr);
-               mmc_select_voltage(host, host->card->ocr);
                /*
                 * Tell runtime PM core we just powered up the card,
                 * since it still believes the card is powered off.
@@ -1066,7 +1065,6 @@ static int mmc_sdio_resume(struct mmc_host *host)
 static int mmc_sdio_power_restore(struct mmc_host *host)
 {
        int ret;
-       u32 ocr, rocr;
 
        BUG_ON(!host);
        BUG_ON(!host->card);
@@ -1088,28 +1086,17 @@ static int mmc_sdio_power_restore(struct mmc_host *host)
         * for OLPC SD8686 (which expects a [CMD5,5,3,7] init sequence), and
         * harmless in other situations.
         *
-        * With these steps taken, mmc_select_voltage() is also required to
-        * restore the correct voltage setting of the card.
         */
 
        sdio_reset(host);
        mmc_go_idle(host);
        mmc_send_if_cond(host, host->ocr_avail);
 
-       ret = mmc_send_io_op_cond(host, 0, &ocr);
+       ret = mmc_send_io_op_cond(host, 0, NULL);
        if (ret)
                goto out;
 
-       if (host->ocr_avail_sdio)
-               host->ocr_avail = host->ocr_avail_sdio;
-
-       rocr = mmc_select_voltage(host, ocr & ~0x7F);
-       if (!rocr) {
-               ret = -EINVAL;
-               goto out;
-       }
-
-       ret = mmc_sdio_init_card(host, rocr, host->card,
+       ret = mmc_sdio_init_card(host, host->card->ocr, host->card,
                                mmc_card_keep_power(host));
        if (!ret && host->sdio_irqs)
                mmc_signal_sdio_irq(host);