2019-05-27 06:55:01 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2007-05-21 18:23:20 +00:00
|
|
|
/*
|
|
|
|
* linux/drivers/mmc/sdio.c
|
|
|
|
*
|
|
|
|
* Copyright 2006-2007 Pierre Ossman
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/err.h>
|
2010-10-02 11:54:10 +00:00
|
|
|
#include <linux/pm_runtime.h>
|
2007-05-21 18:23:20 +00:00
|
|
|
|
|
|
|
#include <linux/mmc/host.h>
|
|
|
|
#include <linux/mmc/card.h>
|
2012-01-11 19:04:52 +00:00
|
|
|
#include <linux/mmc/mmc.h>
|
2007-06-11 18:25:43 +00:00
|
|
|
#include <linux/mmc/sdio.h>
|
2007-05-26 11:48:18 +00:00
|
|
|
#include <linux/mmc/sdio_func.h>
|
2011-04-05 14:50:14 +00:00
|
|
|
#include <linux/mmc/sdio_ids.h>
|
2007-05-21 18:23:20 +00:00
|
|
|
|
|
|
|
#include "core.h"
|
2017-01-13 13:14:14 +00:00
|
|
|
#include "card.h"
|
2017-01-13 13:14:15 +00:00
|
|
|
#include "host.h"
|
2007-05-21 18:23:20 +00:00
|
|
|
#include "bus.h"
|
2017-02-15 08:35:28 +00:00
|
|
|
#include "quirks.h"
|
2010-08-11 01:01:40 +00:00
|
|
|
#include "sd.h"
|
2007-05-26 11:48:18 +00:00
|
|
|
#include "sdio_bus.h"
|
2007-05-21 18:23:20 +00:00
|
|
|
#include "mmc_ops.h"
|
|
|
|
#include "sd_ops.h"
|
|
|
|
#include "sdio_ops.h"
|
2007-06-16 06:04:16 +00:00
|
|
|
#include "sdio_cis.h"
|
2007-05-21 18:23:20 +00:00
|
|
|
|
2020-05-27 11:08:58 +00:00
|
|
|
MMC_DEV_ATTR(vendor, "0x%04x\n", card->cis.vendor);
|
|
|
|
MMC_DEV_ATTR(device, "0x%04x\n", card->cis.device);
|
2020-07-27 13:38:37 +00:00
|
|
|
MMC_DEV_ATTR(revision, "%u.%u\n", card->major_rev, card->minor_rev);
|
2020-05-27 11:08:58 +00:00
|
|
|
MMC_DEV_ATTR(ocr, "0x%08x\n", card->ocr);
|
|
|
|
MMC_DEV_ATTR(rca, "0x%04x\n", card->rca);
|
|
|
|
|
2020-07-27 13:38:37 +00:00
|
|
|
#define sdio_info_attr(num) \
|
|
|
|
static ssize_t info##num##_show(struct device *dev, struct device_attribute *attr, char *buf) \
|
|
|
|
{ \
|
|
|
|
struct mmc_card *card = mmc_dev_to_card(dev); \
|
|
|
|
\
|
|
|
|
if (num > card->num_info) \
|
|
|
|
return -ENODATA; \
|
|
|
|
if (!card->info[num-1][0]) \
|
|
|
|
return 0; \
|
|
|
|
return sprintf(buf, "%s\n", card->info[num-1]); \
|
|
|
|
} \
|
|
|
|
static DEVICE_ATTR_RO(info##num)
|
|
|
|
|
|
|
|
sdio_info_attr(1);
|
|
|
|
sdio_info_attr(2);
|
|
|
|
sdio_info_attr(3);
|
|
|
|
sdio_info_attr(4);
|
|
|
|
|
2020-05-27 11:08:58 +00:00
|
|
|
static struct attribute *sdio_std_attrs[] = {
|
|
|
|
&dev_attr_vendor.attr,
|
|
|
|
&dev_attr_device.attr,
|
2020-07-27 13:38:37 +00:00
|
|
|
&dev_attr_revision.attr,
|
|
|
|
&dev_attr_info1.attr,
|
|
|
|
&dev_attr_info2.attr,
|
|
|
|
&dev_attr_info3.attr,
|
|
|
|
&dev_attr_info4.attr,
|
2020-05-27 11:08:58 +00:00
|
|
|
&dev_attr_ocr.attr,
|
|
|
|
&dev_attr_rca.attr,
|
|
|
|
NULL,
|
|
|
|
};
|
|
|
|
ATTRIBUTE_GROUPS(sdio_std);
|
|
|
|
|
|
|
|
static struct device_type sdio_type = {
|
|
|
|
.groups = sdio_std_groups,
|
|
|
|
};
|
|
|
|
|
2007-06-11 19:01:00 +00:00
|
|
|
static int sdio_read_fbr(struct sdio_func *func)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
unsigned char data;
|
|
|
|
|
2011-04-05 14:50:14 +00:00
|
|
|
if (mmc_card_nonstd_func_interface(func->card)) {
|
|
|
|
func->class = SDIO_CLASS_NONE;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-06-11 19:01:00 +00:00
|
|
|
ret = mmc_io_rw_direct(func->card, 0, 0,
|
2007-08-08 13:23:05 +00:00
|
|
|
SDIO_FBR_BASE(func->num) + SDIO_FBR_STD_IF, 0, &data);
|
2007-06-11 19:01:00 +00:00
|
|
|
if (ret)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
data &= 0x0f;
|
|
|
|
|
|
|
|
if (data == 0x0f) {
|
|
|
|
ret = mmc_io_rw_direct(func->card, 0, 0,
|
2007-08-08 13:23:05 +00:00
|
|
|
SDIO_FBR_BASE(func->num) + SDIO_FBR_STD_IF_EXT, 0, &data);
|
2007-06-11 19:01:00 +00:00
|
|
|
if (ret)
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
func->class = data;
|
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2007-05-26 11:48:18 +00:00
|
|
|
static int sdio_init_func(struct mmc_card *card, unsigned int fn)
|
|
|
|
{
|
2007-06-11 19:01:00 +00:00
|
|
|
int ret;
|
2007-05-26 11:48:18 +00:00
|
|
|
struct sdio_func *func;
|
|
|
|
|
2016-11-02 07:24:00 +00:00
|
|
|
if (WARN_ON(fn > SDIO_MAX_FUNCS))
|
|
|
|
return -EINVAL;
|
2007-05-26 11:48:18 +00:00
|
|
|
|
|
|
|
func = sdio_alloc_func(card);
|
|
|
|
if (IS_ERR(func))
|
|
|
|
return PTR_ERR(func);
|
|
|
|
|
|
|
|
func->num = fn;
|
|
|
|
|
2010-08-11 01:01:50 +00:00
|
|
|
if (!(card->quirks & MMC_QUIRK_NONSTD_SDIO)) {
|
|
|
|
ret = sdio_read_fbr(func);
|
|
|
|
if (ret)
|
|
|
|
goto fail;
|
2007-06-11 19:01:00 +00:00
|
|
|
|
2010-08-11 01:01:50 +00:00
|
|
|
ret = sdio_read_func_cis(func);
|
|
|
|
if (ret)
|
|
|
|
goto fail;
|
|
|
|
} else {
|
|
|
|
func->vendor = func->card->cis.vendor;
|
|
|
|
func->device = func->card->cis.device;
|
|
|
|
func->max_blksize = func->card->cis.blksize;
|
|
|
|
}
|
2007-06-16 06:04:16 +00:00
|
|
|
|
2007-05-26 11:48:18 +00:00
|
|
|
card->sdio_func[fn - 1] = func;
|
|
|
|
|
|
|
|
return 0;
|
2007-06-11 19:01:00 +00:00
|
|
|
|
|
|
|
fail:
|
|
|
|
/*
|
|
|
|
* It is okay to remove the function here even though we hold
|
|
|
|
* the host lock as we haven't registered the device yet.
|
|
|
|
*/
|
|
|
|
sdio_remove_func(func);
|
|
|
|
return ret;
|
2007-05-26 11:48:18 +00:00
|
|
|
}
|
|
|
|
|
2012-01-26 14:57:10 +00:00
|
|
|
static int sdio_read_cccr(struct mmc_card *card, u32 ocr)
|
2007-06-11 18:25:43 +00:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
int cccr_vsn;
|
2012-01-26 14:57:10 +00:00
|
|
|
int uhs = ocr & R4_18V_PRESENT;
|
2007-06-11 18:25:43 +00:00
|
|
|
unsigned char data;
|
2011-11-15 03:14:38 +00:00
|
|
|
unsigned char speed;
|
2007-06-11 18:25:43 +00:00
|
|
|
|
|
|
|
ret = mmc_io_rw_direct(card, 0, 0, SDIO_CCCR_CCCR, 0, &data);
|
|
|
|
if (ret)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
cccr_vsn = data & 0x0f;
|
|
|
|
|
2011-10-21 02:16:32 +00:00
|
|
|
if (cccr_vsn > SDIO_CCCR_REV_3_00) {
|
2011-10-11 06:14:09 +00:00
|
|
|
pr_err("%s: unrecognised CCCR structure version %d\n",
|
2007-06-11 18:25:43 +00:00
|
|
|
mmc_hostname(card->host), cccr_vsn);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
card->cccr.sdio_vsn = (data & 0xf0) >> 4;
|
|
|
|
|
|
|
|
ret = mmc_io_rw_direct(card, 0, 0, SDIO_CCCR_CAPS, 0, &data);
|
|
|
|
if (ret)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (data & SDIO_CCCR_CAP_SMB)
|
|
|
|
card->cccr.multi_block = 1;
|
|
|
|
if (data & SDIO_CCCR_CAP_LSC)
|
|
|
|
card->cccr.low_speed = 1;
|
|
|
|
if (data & SDIO_CCCR_CAP_4BLS)
|
|
|
|
card->cccr.wide_bus = 1;
|
|
|
|
|
|
|
|
if (cccr_vsn >= SDIO_CCCR_REV_1_10) {
|
|
|
|
ret = mmc_io_rw_direct(card, 0, 0, SDIO_CCCR_POWER, 0, &data);
|
|
|
|
if (ret)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (data & SDIO_POWER_SMPC)
|
|
|
|
card->cccr.high_power = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cccr_vsn >= SDIO_CCCR_REV_1_20) {
|
2011-11-15 03:14:38 +00:00
|
|
|
ret = mmc_io_rw_direct(card, 0, 0, SDIO_CCCR_SPEED, 0, &speed);
|
2007-06-11 18:25:43 +00:00
|
|
|
if (ret)
|
|
|
|
goto out;
|
|
|
|
|
2011-11-15 03:14:38 +00:00
|
|
|
card->scr.sda_spec3 = 0;
|
|
|
|
card->sw_caps.sd3_bus_mode = 0;
|
|
|
|
card->sw_caps.sd3_drv_type = 0;
|
2012-01-26 14:57:10 +00:00
|
|
|
if (cccr_vsn >= SDIO_CCCR_REV_3_00 && uhs) {
|
2011-11-15 03:14:38 +00:00
|
|
|
card->scr.sda_spec3 = 1;
|
|
|
|
ret = mmc_io_rw_direct(card, 0, 0,
|
|
|
|
SDIO_CCCR_UHS, 0, &data);
|
|
|
|
if (ret)
|
|
|
|
goto out;
|
|
|
|
|
2012-12-04 11:36:19 +00:00
|
|
|
if (mmc_host_uhs(card->host)) {
|
2011-11-15 03:14:38 +00:00
|
|
|
if (data & SDIO_UHS_DDR50)
|
|
|
|
card->sw_caps.sd3_bus_mode
|
2020-06-22 08:19:19 +00:00
|
|
|
|= SD_MODE_UHS_DDR50 | SD_MODE_UHS_SDR50
|
|
|
|
| SD_MODE_UHS_SDR25 | SD_MODE_UHS_SDR12;
|
2011-11-15 03:14:38 +00:00
|
|
|
|
|
|
|
if (data & SDIO_UHS_SDR50)
|
|
|
|
card->sw_caps.sd3_bus_mode
|
2020-06-22 08:19:19 +00:00
|
|
|
|= SD_MODE_UHS_SDR50 | SD_MODE_UHS_SDR25
|
|
|
|
| SD_MODE_UHS_SDR12;
|
2011-11-15 03:14:38 +00:00
|
|
|
|
|
|
|
if (data & SDIO_UHS_SDR104)
|
|
|
|
card->sw_caps.sd3_bus_mode
|
2020-06-22 08:19:19 +00:00
|
|
|
|= SD_MODE_UHS_SDR104 | SD_MODE_UHS_SDR50
|
|
|
|
| SD_MODE_UHS_SDR25 | SD_MODE_UHS_SDR12;
|
2011-11-15 03:14:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = mmc_io_rw_direct(card, 0, 0,
|
|
|
|
SDIO_CCCR_DRIVE_STRENGTH, 0, &data);
|
|
|
|
if (ret)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (data & SDIO_DRIVE_SDTA)
|
|
|
|
card->sw_caps.sd3_drv_type |= SD_DRIVER_TYPE_A;
|
|
|
|
if (data & SDIO_DRIVE_SDTC)
|
|
|
|
card->sw_caps.sd3_drv_type |= SD_DRIVER_TYPE_C;
|
|
|
|
if (data & SDIO_DRIVE_SDTD)
|
|
|
|
card->sw_caps.sd3_drv_type |= SD_DRIVER_TYPE_D;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if no uhs mode ensure we check for high speed */
|
|
|
|
if (!card->sw_caps.sd3_bus_mode) {
|
|
|
|
if (speed & SDIO_SPEED_SHS) {
|
|
|
|
card->cccr.high_speed = 1;
|
|
|
|
card->sw_caps.hs_max_dtr = 50000000;
|
|
|
|
} else {
|
|
|
|
card->cccr.high_speed = 0;
|
|
|
|
card->sw_caps.hs_max_dtr = 25000000;
|
|
|
|
}
|
|
|
|
}
|
2007-06-11 18:25:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2007-07-30 16:23:53 +00:00
|
|
|
static int sdio_enable_wide(struct mmc_card *card)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
u8 ctrl;
|
|
|
|
|
|
|
|
if (!(card->host->caps & MMC_CAP_4_BIT_DATA))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (card->cccr.low_speed && !card->cccr.wide_bus)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
ret = mmc_io_rw_direct(card, 0, 0, SDIO_CCCR_IF, 0, &ctrl);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2012-05-15 05:09:43 +00:00
|
|
|
if ((ctrl & SDIO_BUS_WIDTH_MASK) == SDIO_BUS_WIDTH_RESERVED)
|
2014-09-12 21:56:56 +00:00
|
|
|
pr_warn("%s: SDIO_CCCR_IF is invalid: 0x%02x\n",
|
|
|
|
mmc_hostname(card->host), ctrl);
|
2012-05-15 05:09:43 +00:00
|
|
|
|
|
|
|
/* set as 4-bit bus width */
|
|
|
|
ctrl &= ~SDIO_BUS_WIDTH_MASK;
|
2007-07-30 16:23:53 +00:00
|
|
|
ctrl |= SDIO_BUS_WIDTH_4BIT;
|
|
|
|
|
|
|
|
ret = mmc_io_rw_direct(card, 1, 0, SDIO_CCCR_IF, ctrl, NULL);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2010-08-11 01:01:40 +00:00
|
|
|
return 1;
|
2007-07-30 16:23:53 +00:00
|
|
|
}
|
|
|
|
|
2009-09-22 23:45:07 +00:00
|
|
|
/*
|
|
|
|
* If desired, disconnect the pull-up resistor on CD/DAT[3] (pin 1)
|
|
|
|
* of the card. This may be required on certain setups of boards,
|
|
|
|
* controllers and embedded sdio device which do not need the card's
|
|
|
|
* pull-up. As a result, card detection is disabled and power is saved.
|
|
|
|
*/
|
|
|
|
static int sdio_disable_cd(struct mmc_card *card)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
u8 ctrl;
|
|
|
|
|
2011-04-05 15:02:25 +00:00
|
|
|
if (!mmc_card_disable_cd(card))
|
2009-09-22 23:45:07 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
ret = mmc_io_rw_direct(card, 0, 0, SDIO_CCCR_IF, 0, &ctrl);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ctrl |= SDIO_BUS_CD_DISABLE;
|
|
|
|
|
|
|
|
return mmc_io_rw_direct(card, 1, 0, SDIO_CCCR_IF, ctrl, NULL);
|
|
|
|
}
|
|
|
|
|
2010-03-05 21:43:34 +00:00
|
|
|
/*
|
|
|
|
* Devices that remain active during a system suspend are
|
|
|
|
* put back into 1-bit mode.
|
|
|
|
*/
|
|
|
|
static int sdio_disable_wide(struct mmc_card *card)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
u8 ctrl;
|
|
|
|
|
|
|
|
if (!(card->host->caps & MMC_CAP_4_BIT_DATA))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (card->cccr.low_speed && !card->cccr.wide_bus)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
ret = mmc_io_rw_direct(card, 0, 0, SDIO_CCCR_IF, 0, &ctrl);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
if (!(ctrl & SDIO_BUS_WIDTH_4BIT))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
ctrl &= ~SDIO_BUS_WIDTH_4BIT;
|
|
|
|
ctrl |= SDIO_BUS_ASYNC_INT;
|
|
|
|
|
|
|
|
ret = mmc_io_rw_direct(card, 1, 0, SDIO_CCCR_IF, ctrl, NULL);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
mmc_set_bus_width(card->host, MMC_BUS_WIDTH_1);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-06-09 08:14:31 +00:00
|
|
|
static int sdio_disable_4bit_bus(struct mmc_card *card)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
|
|
|
if (card->type == MMC_TYPE_SDIO)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (!(card->host->caps & MMC_CAP_4_BIT_DATA))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (!(card->scr.bus_widths & SD_SCR_BUS_WIDTH_4))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
err = mmc_app_set_bus_width(card, MMC_BUS_WIDTH_1);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
out:
|
|
|
|
return sdio_disable_wide(card);
|
|
|
|
}
|
|
|
|
|
2010-08-11 01:01:40 +00:00
|
|
|
|
|
|
|
static int sdio_enable_4bit_bus(struct mmc_card *card)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
2020-06-08 10:30:09 +00:00
|
|
|
err = sdio_enable_wide(card);
|
|
|
|
if (err <= 0)
|
|
|
|
return err;
|
2010-08-11 01:01:40 +00:00
|
|
|
if (card->type == MMC_TYPE_SDIO)
|
2020-06-08 10:30:09 +00:00
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (card->scr.bus_widths & SD_SCR_BUS_WIDTH_4) {
|
2010-08-11 01:01:40 +00:00
|
|
|
err = mmc_app_set_bus_width(card, MMC_BUS_WIDTH_4);
|
2020-06-08 10:30:09 +00:00
|
|
|
if (err) {
|
|
|
|
sdio_disable_wide(card);
|
2010-08-11 01:01:40 +00:00
|
|
|
return err;
|
2020-06-08 10:30:09 +00:00
|
|
|
}
|
2015-02-24 02:42:23 +00:00
|
|
|
}
|
2020-06-08 10:30:09 +00:00
|
|
|
out:
|
|
|
|
mmc_set_bus_width(card->host, MMC_BUS_WIDTH_4);
|
2010-08-11 01:01:40 +00:00
|
|
|
|
2020-06-08 10:30:09 +00:00
|
|
|
return 0;
|
2010-08-11 01:01:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-08-31 15:22:46 +00:00
|
|
|
/*
|
|
|
|
* Test if the card supports high-speed mode and, if so, switch to it.
|
|
|
|
*/
|
2010-08-11 01:01:40 +00:00
|
|
|
static int mmc_sdio_switch_hs(struct mmc_card *card, int enable)
|
2008-08-31 15:22:46 +00:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
u8 speed;
|
|
|
|
|
|
|
|
if (!(card->host->caps & MMC_CAP_SD_HIGHSPEED))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (!card->cccr.high_speed)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
ret = mmc_io_rw_direct(card, 0, 0, SDIO_CCCR_SPEED, 0, &speed);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2010-08-11 01:01:40 +00:00
|
|
|
if (enable)
|
|
|
|
speed |= SDIO_SPEED_EHS;
|
|
|
|
else
|
|
|
|
speed &= ~SDIO_SPEED_EHS;
|
2008-08-31 15:22:46 +00:00
|
|
|
|
|
|
|
ret = mmc_io_rw_direct(card, 1, 0, SDIO_CCCR_SPEED, speed, NULL);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2010-08-11 01:01:40 +00:00
|
|
|
return 1;
|
|
|
|
}
|
2008-08-31 15:22:46 +00:00
|
|
|
|
2010-08-11 01:01:40 +00:00
|
|
|
/*
|
|
|
|
* Enable SDIO/combo card's high-speed mode. Return 0/1 if [not]supported.
|
|
|
|
*/
|
|
|
|
static int sdio_enable_hs(struct mmc_card *card)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = mmc_sdio_switch_hs(card, true);
|
|
|
|
if (ret <= 0 || card->type == MMC_TYPE_SDIO)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ret = mmc_sd_switch_hs(card);
|
|
|
|
if (ret <= 0)
|
|
|
|
mmc_sdio_switch_hs(card, false);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-08-11 01:01:40 +00:00
|
|
|
static unsigned mmc_sdio_get_max_clock(struct mmc_card *card)
|
|
|
|
{
|
|
|
|
unsigned max_dtr;
|
|
|
|
|
2014-04-23 08:07:35 +00:00
|
|
|
if (mmc_card_hs(card)) {
|
2010-08-11 01:01:40 +00:00
|
|
|
/*
|
|
|
|
* The SDIO specification doesn't mention how
|
|
|
|
* the CIS transfer speed register relates to
|
|
|
|
* high-speed, but it seems that 50 MHz is
|
|
|
|
* mandatory.
|
|
|
|
*/
|
|
|
|
max_dtr = 50000000;
|
|
|
|
} else {
|
|
|
|
max_dtr = card->cis.max_dtr;
|
|
|
|
}
|
|
|
|
|
2010-08-11 01:01:40 +00:00
|
|
|
if (card->type == MMC_TYPE_SD_COMBO)
|
|
|
|
max_dtr = min(max_dtr, mmc_sd_get_max_clock(card));
|
|
|
|
|
2010-08-11 01:01:40 +00:00
|
|
|
return max_dtr;
|
2008-08-31 15:22:46 +00:00
|
|
|
}
|
|
|
|
|
2011-11-15 03:14:38 +00:00
|
|
|
static unsigned char host_drive_to_sdio_drive(int host_strength)
|
|
|
|
{
|
|
|
|
switch (host_strength) {
|
|
|
|
case MMC_SET_DRIVER_TYPE_A:
|
|
|
|
return SDIO_DTSx_SET_TYPE_A;
|
|
|
|
case MMC_SET_DRIVER_TYPE_B:
|
|
|
|
return SDIO_DTSx_SET_TYPE_B;
|
|
|
|
case MMC_SET_DRIVER_TYPE_C:
|
|
|
|
return SDIO_DTSx_SET_TYPE_C;
|
|
|
|
case MMC_SET_DRIVER_TYPE_D:
|
|
|
|
return SDIO_DTSx_SET_TYPE_D;
|
|
|
|
default:
|
|
|
|
return SDIO_DTSx_SET_TYPE_B;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sdio_select_driver_type(struct mmc_card *card)
|
|
|
|
{
|
2015-02-06 12:12:53 +00:00
|
|
|
int card_drv_type, drive_strength, drv_type;
|
2011-11-15 03:14:38 +00:00
|
|
|
unsigned char card_strength;
|
|
|
|
int err;
|
|
|
|
|
2015-02-06 12:12:56 +00:00
|
|
|
card->drive_strength = 0;
|
|
|
|
|
2015-02-06 12:12:53 +00:00
|
|
|
card_drv_type = card->sw_caps.sd3_drv_type | SD_DRIVER_TYPE_B;
|
2011-11-15 03:14:38 +00:00
|
|
|
|
2015-02-06 12:12:55 +00:00
|
|
|
drive_strength = mmc_select_drive_strength(card,
|
|
|
|
card->sw_caps.uhs_max_dtr,
|
|
|
|
card_drv_type, &drv_type);
|
2011-11-15 03:14:38 +00:00
|
|
|
|
2015-02-06 12:12:52 +00:00
|
|
|
if (drive_strength) {
|
|
|
|
/* if error just use default for drive strength B */
|
|
|
|
err = mmc_io_rw_direct(card, 0, 0, SDIO_CCCR_DRIVE_STRENGTH, 0,
|
|
|
|
&card_strength);
|
|
|
|
if (err)
|
|
|
|
return;
|
2011-11-15 03:14:38 +00:00
|
|
|
|
2015-02-06 12:12:52 +00:00
|
|
|
card_strength &= ~(SDIO_DRIVE_DTSx_MASK<<SDIO_DRIVE_DTSx_SHIFT);
|
|
|
|
card_strength |= host_drive_to_sdio_drive(drive_strength);
|
2011-11-15 03:14:38 +00:00
|
|
|
|
2015-02-06 12:12:52 +00:00
|
|
|
/* if error default to drive strength B */
|
|
|
|
err = mmc_io_rw_direct(card, 1, 0, SDIO_CCCR_DRIVE_STRENGTH,
|
|
|
|
card_strength, NULL);
|
|
|
|
if (err)
|
|
|
|
return;
|
2015-02-06 12:12:56 +00:00
|
|
|
card->drive_strength = drive_strength;
|
2015-02-06 12:12:52 +00:00
|
|
|
}
|
2011-11-15 03:14:38 +00:00
|
|
|
|
2015-02-06 12:12:52 +00:00
|
|
|
if (drv_type)
|
|
|
|
mmc_set_driver_type(card->host, drv_type);
|
2011-11-15 03:14:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int sdio_set_bus_speed_mode(struct mmc_card *card)
|
|
|
|
{
|
|
|
|
unsigned int bus_speed, timing;
|
|
|
|
int err;
|
|
|
|
unsigned char speed;
|
2018-04-10 07:00:31 +00:00
|
|
|
unsigned int max_rate;
|
2011-11-15 03:14:38 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* If the host doesn't support any of the UHS-I modes, fallback on
|
|
|
|
* default speed.
|
|
|
|
*/
|
2012-12-04 11:36:19 +00:00
|
|
|
if (!mmc_host_uhs(card->host))
|
2011-11-15 03:14:38 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
bus_speed = SDIO_SPEED_SDR12;
|
|
|
|
timing = MMC_TIMING_UHS_SDR12;
|
|
|
|
if ((card->host->caps & MMC_CAP_UHS_SDR104) &&
|
|
|
|
(card->sw_caps.sd3_bus_mode & SD_MODE_UHS_SDR104)) {
|
|
|
|
bus_speed = SDIO_SPEED_SDR104;
|
|
|
|
timing = MMC_TIMING_UHS_SDR104;
|
|
|
|
card->sw_caps.uhs_max_dtr = UHS_SDR104_MAX_DTR;
|
2012-12-04 11:36:20 +00:00
|
|
|
card->sd_bus_speed = UHS_SDR104_BUS_SPEED;
|
2011-11-15 03:14:38 +00:00
|
|
|
} else if ((card->host->caps & MMC_CAP_UHS_DDR50) &&
|
|
|
|
(card->sw_caps.sd3_bus_mode & SD_MODE_UHS_DDR50)) {
|
|
|
|
bus_speed = SDIO_SPEED_DDR50;
|
|
|
|
timing = MMC_TIMING_UHS_DDR50;
|
|
|
|
card->sw_caps.uhs_max_dtr = UHS_DDR50_MAX_DTR;
|
2012-12-04 11:36:20 +00:00
|
|
|
card->sd_bus_speed = UHS_DDR50_BUS_SPEED;
|
2011-11-15 03:14:38 +00:00
|
|
|
} else if ((card->host->caps & (MMC_CAP_UHS_SDR104 |
|
|
|
|
MMC_CAP_UHS_SDR50)) && (card->sw_caps.sd3_bus_mode &
|
|
|
|
SD_MODE_UHS_SDR50)) {
|
|
|
|
bus_speed = SDIO_SPEED_SDR50;
|
|
|
|
timing = MMC_TIMING_UHS_SDR50;
|
|
|
|
card->sw_caps.uhs_max_dtr = UHS_SDR50_MAX_DTR;
|
2012-12-04 11:36:20 +00:00
|
|
|
card->sd_bus_speed = UHS_SDR50_BUS_SPEED;
|
2011-11-15 03:14:38 +00:00
|
|
|
} else if ((card->host->caps & (MMC_CAP_UHS_SDR104 |
|
|
|
|
MMC_CAP_UHS_SDR50 | MMC_CAP_UHS_SDR25)) &&
|
|
|
|
(card->sw_caps.sd3_bus_mode & SD_MODE_UHS_SDR25)) {
|
|
|
|
bus_speed = SDIO_SPEED_SDR25;
|
|
|
|
timing = MMC_TIMING_UHS_SDR25;
|
|
|
|
card->sw_caps.uhs_max_dtr = UHS_SDR25_MAX_DTR;
|
2012-12-04 11:36:20 +00:00
|
|
|
card->sd_bus_speed = UHS_SDR25_BUS_SPEED;
|
2011-11-15 03:14:38 +00:00
|
|
|
} else if ((card->host->caps & (MMC_CAP_UHS_SDR104 |
|
|
|
|
MMC_CAP_UHS_SDR50 | MMC_CAP_UHS_SDR25 |
|
|
|
|
MMC_CAP_UHS_SDR12)) && (card->sw_caps.sd3_bus_mode &
|
|
|
|
SD_MODE_UHS_SDR12)) {
|
|
|
|
bus_speed = SDIO_SPEED_SDR12;
|
|
|
|
timing = MMC_TIMING_UHS_SDR12;
|
|
|
|
card->sw_caps.uhs_max_dtr = UHS_SDR12_MAX_DTR;
|
2012-12-04 11:36:20 +00:00
|
|
|
card->sd_bus_speed = UHS_SDR12_BUS_SPEED;
|
2011-11-15 03:14:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
err = mmc_io_rw_direct(card, 0, 0, SDIO_CCCR_SPEED, 0, &speed);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
speed &= ~SDIO_SPEED_BSS_MASK;
|
|
|
|
speed |= bus_speed;
|
|
|
|
err = mmc_io_rw_direct(card, 1, 0, SDIO_CCCR_SPEED, speed, NULL);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
2018-04-10 07:00:31 +00:00
|
|
|
max_rate = min_not_zero(card->quirk_max_rate,
|
|
|
|
card->sw_caps.uhs_max_dtr);
|
|
|
|
|
2020-06-22 08:19:19 +00:00
|
|
|
mmc_set_timing(card->host, timing);
|
|
|
|
mmc_set_clock(card->host, max_rate);
|
2011-11-15 03:14:38 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* UHS-I specific initialization procedure
|
|
|
|
*/
|
|
|
|
static int mmc_sdio_init_uhs_card(struct mmc_card *card)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
|
|
|
if (!card->scr.sda_spec3)
|
|
|
|
return 0;
|
|
|
|
|
2018-02-27 01:26:34 +00:00
|
|
|
/* Switch to wider bus */
|
|
|
|
err = sdio_enable_4bit_bus(card);
|
|
|
|
if (err)
|
|
|
|
goto out;
|
2011-11-15 03:14:38 +00:00
|
|
|
|
|
|
|
/* Set the driver strength for the card */
|
|
|
|
sdio_select_driver_type(card);
|
|
|
|
|
|
|
|
/* Set bus speed mode of the card */
|
|
|
|
err = sdio_set_bus_speed_mode(card);
|
|
|
|
if (err)
|
|
|
|
goto out;
|
|
|
|
|
2013-04-17 11:50:53 +00:00
|
|
|
/*
|
|
|
|
* SPI mode doesn't define CMD19 and tuning is only valid for SDR50 and
|
|
|
|
* SDR104 mode SD-cards. Note that tuning is mandatory for SDR104.
|
|
|
|
*/
|
2014-12-05 17:40:59 +00:00
|
|
|
if (!mmc_host_is_spi(card->host) &&
|
2016-01-13 08:36:55 +00:00
|
|
|
((card->host->ios.timing == MMC_TIMING_UHS_SDR50) ||
|
|
|
|
(card->host->ios.timing == MMC_TIMING_UHS_SDR104)))
|
2014-12-05 17:40:59 +00:00
|
|
|
err = mmc_execute_tuning(card);
|
2011-11-15 03:14:38 +00:00
|
|
|
out:
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-04-30 09:16:40 +00:00
|
|
|
static int mmc_sdio_pre_init(struct mmc_host *host, u32 ocr,
|
|
|
|
struct mmc_card *card)
|
2017-01-09 08:56:19 +00:00
|
|
|
{
|
2020-04-30 09:16:40 +00:00
|
|
|
if (card)
|
|
|
|
mmc_remove_card(card);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Reset the card by performing the same steps that are taken by
|
|
|
|
* mmc_rescan_try_freq() and mmc_attach_sdio() during a "normal" probe.
|
|
|
|
*
|
|
|
|
* sdio_reset() is technically not needed. Having just powered up the
|
|
|
|
* hardware, it should already be in reset state. However, some
|
|
|
|
* platforms (such as SD8686 on OLPC) do not instantly cut power,
|
|
|
|
* meaning that a reset is required when restoring power soon after
|
|
|
|
* powering off. It is harmless in other cases.
|
|
|
|
*
|
|
|
|
* The CMD5 reset (mmc_send_io_op_cond()), according to the SDIO spec,
|
|
|
|
* is not necessary for non-removable cards. However, it is required
|
|
|
|
* for OLPC SD8686 (which expects a [CMD5,5,3,7] init sequence), and
|
|
|
|
* harmless in other situations.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2017-01-09 08:56:19 +00:00
|
|
|
sdio_reset(host);
|
|
|
|
mmc_go_idle(host);
|
2020-04-30 09:16:39 +00:00
|
|
|
mmc_send_if_cond(host, ocr);
|
2020-04-30 09:16:40 +00:00
|
|
|
return mmc_send_io_op_cond(host, 0, NULL);
|
2017-01-09 08:56:19 +00:00
|
|
|
}
|
|
|
|
|
2009-09-22 23:45:28 +00:00
|
|
|
/*
|
|
|
|
* Handle the detection and initialisation of a card.
|
|
|
|
*
|
|
|
|
* In the case of a resume, "oldcard" will contain the card
|
|
|
|
* we're trying to reinitialise.
|
|
|
|
*/
|
|
|
|
static int mmc_sdio_init_card(struct mmc_host *host, u32 ocr,
|
2019-06-17 23:20:50 +00:00
|
|
|
struct mmc_card *oldcard)
|
2009-09-22 23:45:28 +00:00
|
|
|
{
|
|
|
|
struct mmc_card *card;
|
|
|
|
int err;
|
2013-01-28 14:08:28 +00:00
|
|
|
int retries = 10;
|
2013-09-13 08:49:34 +00:00
|
|
|
u32 rocr = 0;
|
2013-09-13 09:31:33 +00:00
|
|
|
u32 ocr_card = ocr;
|
2009-09-22 23:45:28 +00:00
|
|
|
|
|
|
|
WARN_ON(!host->claimed);
|
|
|
|
|
2013-09-13 08:49:34 +00:00
|
|
|
/* to query card if 1.8V signalling is supported */
|
|
|
|
if (mmc_host_uhs(host))
|
|
|
|
ocr |= R4_18V_PRESENT;
|
|
|
|
|
2013-01-28 14:08:28 +00:00
|
|
|
try_again:
|
|
|
|
if (!retries) {
|
2014-09-12 21:56:56 +00:00
|
|
|
pr_warn("%s: Skipping voltage switch\n", mmc_hostname(host));
|
2013-01-28 14:08:28 +00:00
|
|
|
ocr &= ~R4_18V_PRESENT;
|
|
|
|
}
|
|
|
|
|
2009-09-22 23:45:28 +00:00
|
|
|
/*
|
|
|
|
* Inform the card of the voltage
|
|
|
|
*/
|
2019-06-17 23:20:50 +00:00
|
|
|
err = mmc_send_io_op_cond(host, ocr, &rocr);
|
|
|
|
if (err)
|
2020-04-30 09:16:38 +00:00
|
|
|
return err;
|
2009-09-22 23:45:28 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* For SPI, enable CRC as appropriate.
|
|
|
|
*/
|
|
|
|
if (mmc_host_is_spi(host)) {
|
|
|
|
err = mmc_spi_set_crc(host, use_spi_crc);
|
|
|
|
if (err)
|
2020-04-30 09:16:38 +00:00
|
|
|
return err;
|
2009-09-22 23:45:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Allocate card structure.
|
|
|
|
*/
|
2020-05-27 11:08:58 +00:00
|
|
|
card = mmc_alloc_card(host, &sdio_type);
|
2020-04-30 09:16:38 +00:00
|
|
|
if (IS_ERR(card))
|
|
|
|
return PTR_ERR(card);
|
2009-09-22 23:45:28 +00:00
|
|
|
|
2013-09-13 08:49:34 +00:00
|
|
|
if ((rocr & R4_MEMORY_PRESENT) &&
|
|
|
|
mmc_sd_get_cid(host, ocr & rocr, card->raw_cid, NULL) == 0) {
|
2010-08-11 01:01:40 +00:00
|
|
|
card->type = MMC_TYPE_SD_COMBO;
|
|
|
|
|
|
|
|
if (oldcard && (oldcard->type != MMC_TYPE_SD_COMBO ||
|
|
|
|
memcmp(card->raw_cid, oldcard->raw_cid, sizeof(card->raw_cid)) != 0)) {
|
2020-04-30 09:16:38 +00:00
|
|
|
err = -ENOENT;
|
|
|
|
goto mismatch;
|
2010-08-11 01:01:40 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
card->type = MMC_TYPE_SDIO;
|
|
|
|
|
|
|
|
if (oldcard && oldcard->type != MMC_TYPE_SDIO) {
|
2020-04-30 09:16:38 +00:00
|
|
|
err = -ENOENT;
|
|
|
|
goto mismatch;
|
2010-08-11 01:01:40 +00:00
|
|
|
}
|
|
|
|
}
|
2009-09-22 23:45:28 +00:00
|
|
|
|
2010-04-01 08:03:25 +00:00
|
|
|
/*
|
|
|
|
* Call the optional HC's init_card function to handle quirks.
|
|
|
|
*/
|
|
|
|
if (host->ops->init_card)
|
|
|
|
host->ops->init_card(host, card);
|
|
|
|
|
2011-11-15 03:14:38 +00:00
|
|
|
/*
|
|
|
|
* If the host and card support UHS-I mode request the card
|
|
|
|
* to switch to 1.8V signaling level. No 1.8v signalling if
|
2012-08-05 14:25:40 +00:00
|
|
|
* UHS mode is not enabled to maintain compatibility and some
|
2011-11-15 03:14:38 +00:00
|
|
|
* systems that claim 1.8v signalling in fact do not support
|
2017-01-09 08:56:20 +00:00
|
|
|
* it. Per SDIO spec v3, section 3.1.2, if the voltage is already
|
|
|
|
* 1.8v, the card sets S18A to 0 in the R4 response. So it will
|
|
|
|
* fails to check rocr & R4_18V_PRESENT, but we still need to
|
|
|
|
* try to init uhs card. sdio_read_cccr will take over this task
|
|
|
|
* to make sure which speed mode should work.
|
2011-11-15 03:14:38 +00:00
|
|
|
*/
|
2019-06-17 23:20:50 +00:00
|
|
|
if (rocr & ocr & R4_18V_PRESENT) {
|
2017-01-25 09:25:01 +00:00
|
|
|
err = mmc_set_uhs_voltage(host, ocr_card);
|
2013-01-28 14:08:28 +00:00
|
|
|
if (err == -EAGAIN) {
|
2020-04-30 09:16:40 +00:00
|
|
|
mmc_sdio_pre_init(host, ocr_card, card);
|
2013-01-28 14:08:28 +00:00
|
|
|
retries--;
|
|
|
|
goto try_again;
|
|
|
|
} else if (err) {
|
2011-11-15 03:14:38 +00:00
|
|
|
ocr &= ~R4_18V_PRESENT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-22 23:45:28 +00:00
|
|
|
/*
|
|
|
|
* For native busses: set card RCA and quit open drain mode.
|
|
|
|
*/
|
2019-06-17 23:20:50 +00:00
|
|
|
if (!mmc_host_is_spi(host)) {
|
2009-09-22 23:45:28 +00:00
|
|
|
err = mmc_send_relative_addr(host, &card->rca);
|
|
|
|
if (err)
|
|
|
|
goto remove;
|
|
|
|
|
2011-03-01 13:41:04 +00:00
|
|
|
/*
|
|
|
|
* Update oldcard with the new RCA received from the SDIO
|
|
|
|
* device -- we're doing this so that it's updated in the
|
|
|
|
* "card" struct when oldcard overwrites that later.
|
|
|
|
*/
|
|
|
|
if (oldcard)
|
|
|
|
oldcard->rca = card->rca;
|
2009-09-22 23:45:28 +00:00
|
|
|
}
|
|
|
|
|
2010-08-11 01:01:40 +00:00
|
|
|
/*
|
|
|
|
* Read CSD, before selecting the card
|
|
|
|
*/
|
|
|
|
if (!oldcard && card->type == MMC_TYPE_SD_COMBO) {
|
2021-01-18 08:45:20 +00:00
|
|
|
err = mmc_sd_get_csd(card);
|
2010-08-11 01:01:40 +00:00
|
|
|
if (err)
|
2020-04-30 09:16:38 +00:00
|
|
|
goto remove;
|
2010-08-11 01:01:40 +00:00
|
|
|
|
|
|
|
mmc_decode_cid(card);
|
|
|
|
}
|
|
|
|
|
2009-09-22 23:45:28 +00:00
|
|
|
/*
|
|
|
|
* Select card, as all following commands rely on that.
|
|
|
|
*/
|
2019-06-17 23:20:50 +00:00
|
|
|
if (!mmc_host_is_spi(host)) {
|
2009-09-22 23:45:28 +00:00
|
|
|
err = mmc_select_card(card);
|
|
|
|
if (err)
|
|
|
|
goto remove;
|
|
|
|
}
|
|
|
|
|
2010-08-11 01:01:50 +00:00
|
|
|
if (card->quirks & MMC_QUIRK_NONSTD_SDIO) {
|
|
|
|
/*
|
|
|
|
* This is non-standard SDIO device, meaning it doesn't
|
|
|
|
* have any CIA (Common I/O area) registers present.
|
|
|
|
* It's host's responsibility to fill cccr and cis
|
|
|
|
* structures in init_card().
|
|
|
|
*/
|
|
|
|
mmc_set_clock(host, card->cis.max_dtr);
|
|
|
|
|
|
|
|
if (card->cccr.high_speed) {
|
|
|
|
mmc_set_timing(card->host, MMC_TIMING_SD_HS);
|
|
|
|
}
|
|
|
|
|
2020-04-30 09:16:38 +00:00
|
|
|
if (oldcard)
|
|
|
|
mmc_remove_card(card);
|
|
|
|
else
|
|
|
|
host->card = card;
|
|
|
|
|
|
|
|
return 0;
|
2010-08-11 01:01:50 +00:00
|
|
|
}
|
|
|
|
|
2009-09-22 23:45:28 +00:00
|
|
|
/*
|
2017-01-09 08:56:20 +00:00
|
|
|
* Read the common registers. Note that we should try to
|
|
|
|
* validate whether UHS would work or not.
|
2009-09-22 23:45:28 +00:00
|
|
|
*/
|
2012-01-26 14:57:10 +00:00
|
|
|
err = sdio_read_cccr(card, ocr);
|
2017-01-09 08:56:20 +00:00
|
|
|
if (err) {
|
2020-04-30 09:16:40 +00:00
|
|
|
mmc_sdio_pre_init(host, ocr_card, card);
|
2017-01-09 08:56:20 +00:00
|
|
|
if (ocr & R4_18V_PRESENT) {
|
|
|
|
/* Retry init sequence, but without R4_18V_PRESENT. */
|
|
|
|
retries = 0;
|
|
|
|
goto try_again;
|
|
|
|
}
|
2020-04-30 09:16:37 +00:00
|
|
|
return err;
|
2017-01-09 08:56:20 +00:00
|
|
|
}
|
2009-09-22 23:45:28 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Read the common CIS tuples.
|
|
|
|
*/
|
|
|
|
err = sdio_read_common_cis(card);
|
|
|
|
if (err)
|
|
|
|
goto remove;
|
|
|
|
|
|
|
|
if (oldcard) {
|
2020-04-30 09:16:38 +00:00
|
|
|
if (card->cis.vendor == oldcard->cis.vendor &&
|
|
|
|
card->cis.device == oldcard->cis.device) {
|
|
|
|
mmc_remove_card(card);
|
|
|
|
card = oldcard;
|
|
|
|
} else {
|
|
|
|
err = -ENOENT;
|
|
|
|
goto mismatch;
|
2019-02-28 06:08:28 +00:00
|
|
|
}
|
2009-09-22 23:45:28 +00:00
|
|
|
}
|
2013-09-13 09:31:33 +00:00
|
|
|
card->ocr = ocr_card;
|
2017-02-15 08:35:30 +00:00
|
|
|
mmc_fixup_device(card, sdio_fixup_methods);
|
2009-09-22 23:45:28 +00:00
|
|
|
|
2010-08-11 01:01:40 +00:00
|
|
|
if (card->type == MMC_TYPE_SD_COMBO) {
|
|
|
|
err = mmc_sd_setup_card(host, card, oldcard != NULL);
|
|
|
|
/* handle as SDIO-only card if memory init failed */
|
|
|
|
if (err) {
|
|
|
|
mmc_go_idle(host);
|
|
|
|
if (mmc_host_is_spi(host))
|
|
|
|
/* should not fail, as it worked previously */
|
|
|
|
mmc_spi_set_crc(host, use_spi_crc);
|
|
|
|
card->type = MMC_TYPE_SDIO;
|
|
|
|
} else
|
|
|
|
card->dev.type = &sd_type;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If needed, disconnect card detection pull-up resistor.
|
|
|
|
*/
|
|
|
|
err = sdio_disable_cd(card);
|
|
|
|
if (err)
|
|
|
|
goto remove;
|
|
|
|
|
2011-11-15 03:14:38 +00:00
|
|
|
/* Initialization sequence for UHS-I cards */
|
|
|
|
/* Only if card supports 1.8v and UHS signaling */
|
|
|
|
if ((ocr & R4_18V_PRESENT) && card->sw_caps.sd3_bus_mode) {
|
|
|
|
err = mmc_sdio_init_uhs_card(card);
|
|
|
|
if (err)
|
|
|
|
goto remove;
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* Switch to high-speed (if supported).
|
|
|
|
*/
|
|
|
|
err = sdio_enable_hs(card);
|
|
|
|
if (err > 0)
|
2014-04-23 08:07:35 +00:00
|
|
|
mmc_set_timing(card->host, MMC_TIMING_SD_HS);
|
2011-11-15 03:14:38 +00:00
|
|
|
else if (err)
|
|
|
|
goto remove;
|
2009-09-22 23:45:28 +00:00
|
|
|
|
2011-11-15 03:14:38 +00:00
|
|
|
/*
|
|
|
|
* Change to the card's maximum speed.
|
|
|
|
*/
|
|
|
|
mmc_set_clock(host, mmc_sdio_get_max_clock(card));
|
2009-09-22 23:45:28 +00:00
|
|
|
|
2011-11-15 03:14:38 +00:00
|
|
|
/*
|
|
|
|
* Switch to wider bus (if supported).
|
|
|
|
*/
|
|
|
|
err = sdio_enable_4bit_bus(card);
|
2015-02-24 02:42:23 +00:00
|
|
|
if (err)
|
2011-11-15 03:14:38 +00:00
|
|
|
goto remove;
|
|
|
|
}
|
2018-04-13 21:54:57 +00:00
|
|
|
|
|
|
|
if (host->caps2 & MMC_CAP2_AVOID_3_3V &&
|
|
|
|
host->ios.signal_voltage == MMC_SIGNAL_VOLTAGE_330) {
|
|
|
|
pr_err("%s: Host failed to negotiate down from 3.3V\n",
|
|
|
|
mmc_hostname(host));
|
|
|
|
err = -EINVAL;
|
|
|
|
goto remove;
|
|
|
|
}
|
2020-04-30 09:16:38 +00:00
|
|
|
|
|
|
|
host->card = card;
|
2009-09-22 23:45:28 +00:00
|
|
|
return 0;
|
|
|
|
|
2020-04-30 09:16:38 +00:00
|
|
|
mismatch:
|
|
|
|
pr_debug("%s: Perhaps the card was replaced\n", mmc_hostname(host));
|
2009-09-22 23:45:28 +00:00
|
|
|
remove:
|
2020-04-30 09:16:38 +00:00
|
|
|
if (oldcard != card)
|
2009-09-22 23:45:28 +00:00
|
|
|
mmc_remove_card(card);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2019-06-17 23:13:02 +00:00
|
|
|
static int mmc_sdio_reinit_card(struct mmc_host *host)
|
2015-05-11 14:46:47 +00:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
2020-04-30 09:16:40 +00:00
|
|
|
ret = mmc_sdio_pre_init(host, host->card->ocr, NULL);
|
2015-05-11 14:46:47 +00:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2019-06-17 23:20:50 +00:00
|
|
|
return mmc_sdio_init_card(host, host->card->ocr, host->card);
|
2015-05-11 14:46:47 +00:00
|
|
|
}
|
|
|
|
|
2007-05-21 18:23:20 +00:00
|
|
|
/*
|
|
|
|
* Host is being removed. Free up the current card.
|
|
|
|
*/
|
|
|
|
static void mmc_sdio_remove(struct mmc_host *host)
|
|
|
|
{
|
2007-05-26 11:48:18 +00:00
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0;i < host->card->sdio_funcs;i++) {
|
|
|
|
if (host->card->sdio_func[i]) {
|
|
|
|
sdio_remove_func(host->card->sdio_func[i]);
|
|
|
|
host->card->sdio_func[i] = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-05-21 18:23:20 +00:00
|
|
|
mmc_remove_card(host->card);
|
|
|
|
host->card = NULL;
|
|
|
|
}
|
|
|
|
|
2011-11-28 14:22:00 +00:00
|
|
|
/*
|
|
|
|
* Card detection - card is alive.
|
|
|
|
*/
|
|
|
|
static int mmc_sdio_alive(struct mmc_host *host)
|
|
|
|
{
|
|
|
|
return mmc_select_card(host->card);
|
|
|
|
}
|
|
|
|
|
2007-05-21 18:23:20 +00:00
|
|
|
/*
|
|
|
|
* Card detection callback from host.
|
|
|
|
*/
|
|
|
|
static void mmc_sdio_detect(struct mmc_host *host)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
2010-10-02 11:54:12 +00:00
|
|
|
/* Make sure card is powered before detecting it */
|
2010-11-19 07:29:09 +00:00
|
|
|
if (host->caps & MMC_CAP_POWER_OFF_CARD) {
|
2021-05-20 12:21:44 +00:00
|
|
|
err = pm_runtime_resume_and_get(&host->card->dev);
|
|
|
|
if (err < 0)
|
2010-11-19 07:29:09 +00:00
|
|
|
goto out;
|
|
|
|
}
|
2010-10-02 11:54:12 +00:00
|
|
|
|
2007-05-21 18:23:20 +00:00
|
|
|
mmc_claim_host(host);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Just check if our card has been removed.
|
|
|
|
*/
|
2011-11-28 14:22:00 +00:00
|
|
|
err = _mmc_detect_card_removed(host);
|
2007-05-21 18:23:20 +00:00
|
|
|
|
|
|
|
mmc_release_host(host);
|
|
|
|
|
2010-11-14 10:40:33 +00:00
|
|
|
/*
|
|
|
|
* Tell PM core it's OK to power off the card now.
|
|
|
|
*
|
|
|
|
* The _sync variant is used in order to ensure that the card
|
|
|
|
* is left powered off in case an error occurred, and the card
|
|
|
|
* is going to be removed.
|
|
|
|
*
|
|
|
|
* Since there is no specific reason to believe a new user
|
|
|
|
* is about to show up at this point, the _sync variant is
|
|
|
|
* desirable anyway.
|
|
|
|
*/
|
2010-11-19 07:29:09 +00:00
|
|
|
if (host->caps & MMC_CAP_POWER_OFF_CARD)
|
|
|
|
pm_runtime_put_sync(&host->card->dev);
|
2010-11-14 10:40:33 +00:00
|
|
|
|
2010-10-02 11:54:12 +00:00
|
|
|
out:
|
2007-05-21 18:23:20 +00:00
|
|
|
if (err) {
|
|
|
|
mmc_sdio_remove(host);
|
|
|
|
|
|
|
|
mmc_claim_host(host);
|
|
|
|
mmc_detach_bus(host);
|
2011-09-21 18:08:13 +00:00
|
|
|
mmc_power_off(host);
|
2007-05-21 18:23:20 +00:00
|
|
|
mmc_release_host(host);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-22 23:45:28 +00:00
|
|
|
/*
|
2013-06-10 15:03:37 +00:00
|
|
|
* SDIO pre_suspend. We need to suspend all functions separately.
|
2009-09-22 23:45:28 +00:00
|
|
|
* Therefore all registered functions must have drivers with suspend
|
|
|
|
* and resume methods. Failing that we simply remove the whole card.
|
|
|
|
*/
|
2013-06-10 15:03:37 +00:00
|
|
|
static int mmc_sdio_pre_suspend(struct mmc_host *host)
|
2009-09-22 23:45:28 +00:00
|
|
|
{
|
mmc: core: Fix hanging on I/O during system suspend for removable cards
The mmc core uses a PM notifier to temporarily during system suspend, turn
off the card detection mechanism for removal/insertion of (e)MMC/SD/SDIO
cards. Additionally, the notifier may be used to remove an SDIO card
entirely, if a corresponding SDIO functional driver don't have the system
suspend/resume callbacks assigned. This behaviour has been around for a
very long time.
However, a recent bug report tells us there are problems with this
approach. More precisely, when receiving the PM_SUSPEND_PREPARE
notification, we may end up hanging on I/O to be completed, thus also
preventing the system from getting suspended.
In the end what happens, is that the cancel_delayed_work_sync() in
mmc_pm_notify() ends up waiting for mmc_rescan() to complete - and since
mmc_rescan() wants to claim the host, it needs to wait for the I/O to be
completed first.
Typically, this problem is triggered in Android, if there is ongoing I/O
while the user decides to suspend, resume and then suspend the system
again. This due to that after the resume, an mmc_rescan() work gets punted
to the workqueue, which job is to verify that the card remains inserted
after the system has resumed.
To fix this problem, userspace needs to become frozen to suspend the I/O,
prior to turning off the card detection mechanism. Therefore, let's drop
the PM notifiers for mmc subsystem altogether and rely on the card
detection to be turned off/on as a part of the system_freezable_wq, that we
are already using.
Moreover, to allow and SDIO card to be removed during system suspend, let's
manage this from a ->prepare() callback, assigned at the mmc_host_class
level. In this way, we can use the parent device (the mmc_host_class
device), to remove the card device that is the child, in the
device_prepare() phase.
Reported-by: Kiwoong Kim <kwmad.kim@samsung.com>
Cc: stable@vger.kernel.org # v4.5+
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
Reviewed-by: Linus Walleij <linus.walleij@linaro.org>
Link: https://lore.kernel.org/r/20210310152900.149380-1-ulf.hansson@linaro.org
Reviewed-by: Kiwoong Kim <kwmad.kim@samsung.com>
2021-03-10 15:29:00 +00:00
|
|
|
int i;
|
2009-09-22 23:45:28 +00:00
|
|
|
|
|
|
|
for (i = 0; i < host->card->sdio_funcs; i++) {
|
|
|
|
struct sdio_func *func = host->card->sdio_func[i];
|
|
|
|
if (func && sdio_func_present(func) && func->dev.driver) {
|
|
|
|
const struct dev_pm_ops *pmops = func->dev.driver->pm;
|
mmc: core: Fix hanging on I/O during system suspend for removable cards
The mmc core uses a PM notifier to temporarily during system suspend, turn
off the card detection mechanism for removal/insertion of (e)MMC/SD/SDIO
cards. Additionally, the notifier may be used to remove an SDIO card
entirely, if a corresponding SDIO functional driver don't have the system
suspend/resume callbacks assigned. This behaviour has been around for a
very long time.
However, a recent bug report tells us there are problems with this
approach. More precisely, when receiving the PM_SUSPEND_PREPARE
notification, we may end up hanging on I/O to be completed, thus also
preventing the system from getting suspended.
In the end what happens, is that the cancel_delayed_work_sync() in
mmc_pm_notify() ends up waiting for mmc_rescan() to complete - and since
mmc_rescan() wants to claim the host, it needs to wait for the I/O to be
completed first.
Typically, this problem is triggered in Android, if there is ongoing I/O
while the user decides to suspend, resume and then suspend the system
again. This due to that after the resume, an mmc_rescan() work gets punted
to the workqueue, which job is to verify that the card remains inserted
after the system has resumed.
To fix this problem, userspace needs to become frozen to suspend the I/O,
prior to turning off the card detection mechanism. Therefore, let's drop
the PM notifiers for mmc subsystem altogether and rely on the card
detection to be turned off/on as a part of the system_freezable_wq, that we
are already using.
Moreover, to allow and SDIO card to be removed during system suspend, let's
manage this from a ->prepare() callback, assigned at the mmc_host_class
level. In this way, we can use the parent device (the mmc_host_class
device), to remove the card device that is the child, in the
device_prepare() phase.
Reported-by: Kiwoong Kim <kwmad.kim@samsung.com>
Cc: stable@vger.kernel.org # v4.5+
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
Reviewed-by: Linus Walleij <linus.walleij@linaro.org>
Link: https://lore.kernel.org/r/20210310152900.149380-1-ulf.hansson@linaro.org
Reviewed-by: Kiwoong Kim <kwmad.kim@samsung.com>
2021-03-10 15:29:00 +00:00
|
|
|
if (!pmops || !pmops->suspend || !pmops->resume)
|
2009-09-22 23:45:29 +00:00
|
|
|
/* force removal of entire card in that case */
|
mmc: core: Fix hanging on I/O during system suspend for removable cards
The mmc core uses a PM notifier to temporarily during system suspend, turn
off the card detection mechanism for removal/insertion of (e)MMC/SD/SDIO
cards. Additionally, the notifier may be used to remove an SDIO card
entirely, if a corresponding SDIO functional driver don't have the system
suspend/resume callbacks assigned. This behaviour has been around for a
very long time.
However, a recent bug report tells us there are problems with this
approach. More precisely, when receiving the PM_SUSPEND_PREPARE
notification, we may end up hanging on I/O to be completed, thus also
preventing the system from getting suspended.
In the end what happens, is that the cancel_delayed_work_sync() in
mmc_pm_notify() ends up waiting for mmc_rescan() to complete - and since
mmc_rescan() wants to claim the host, it needs to wait for the I/O to be
completed first.
Typically, this problem is triggered in Android, if there is ongoing I/O
while the user decides to suspend, resume and then suspend the system
again. This due to that after the resume, an mmc_rescan() work gets punted
to the workqueue, which job is to verify that the card remains inserted
after the system has resumed.
To fix this problem, userspace needs to become frozen to suspend the I/O,
prior to turning off the card detection mechanism. Therefore, let's drop
the PM notifiers for mmc subsystem altogether and rely on the card
detection to be turned off/on as a part of the system_freezable_wq, that we
are already using.
Moreover, to allow and SDIO card to be removed during system suspend, let's
manage this from a ->prepare() callback, assigned at the mmc_host_class
level. In this way, we can use the parent device (the mmc_host_class
device), to remove the card device that is the child, in the
device_prepare() phase.
Reported-by: Kiwoong Kim <kwmad.kim@samsung.com>
Cc: stable@vger.kernel.org # v4.5+
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
Reviewed-by: Linus Walleij <linus.walleij@linaro.org>
Link: https://lore.kernel.org/r/20210310152900.149380-1-ulf.hansson@linaro.org
Reviewed-by: Kiwoong Kim <kwmad.kim@samsung.com>
2021-03-10 15:29:00 +00:00
|
|
|
goto remove;
|
2013-06-10 15:03:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
mmc: core: Fix hanging on I/O during system suspend for removable cards
The mmc core uses a PM notifier to temporarily during system suspend, turn
off the card detection mechanism for removal/insertion of (e)MMC/SD/SDIO
cards. Additionally, the notifier may be used to remove an SDIO card
entirely, if a corresponding SDIO functional driver don't have the system
suspend/resume callbacks assigned. This behaviour has been around for a
very long time.
However, a recent bug report tells us there are problems with this
approach. More precisely, when receiving the PM_SUSPEND_PREPARE
notification, we may end up hanging on I/O to be completed, thus also
preventing the system from getting suspended.
In the end what happens, is that the cancel_delayed_work_sync() in
mmc_pm_notify() ends up waiting for mmc_rescan() to complete - and since
mmc_rescan() wants to claim the host, it needs to wait for the I/O to be
completed first.
Typically, this problem is triggered in Android, if there is ongoing I/O
while the user decides to suspend, resume and then suspend the system
again. This due to that after the resume, an mmc_rescan() work gets punted
to the workqueue, which job is to verify that the card remains inserted
after the system has resumed.
To fix this problem, userspace needs to become frozen to suspend the I/O,
prior to turning off the card detection mechanism. Therefore, let's drop
the PM notifiers for mmc subsystem altogether and rely on the card
detection to be turned off/on as a part of the system_freezable_wq, that we
are already using.
Moreover, to allow and SDIO card to be removed during system suspend, let's
manage this from a ->prepare() callback, assigned at the mmc_host_class
level. In this way, we can use the parent device (the mmc_host_class
device), to remove the card device that is the child, in the
device_prepare() phase.
Reported-by: Kiwoong Kim <kwmad.kim@samsung.com>
Cc: stable@vger.kernel.org # v4.5+
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
Reviewed-by: Linus Walleij <linus.walleij@linaro.org>
Link: https://lore.kernel.org/r/20210310152900.149380-1-ulf.hansson@linaro.org
Reviewed-by: Kiwoong Kim <kwmad.kim@samsung.com>
2021-03-10 15:29:00 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
remove:
|
|
|
|
if (!mmc_card_is_removable(host)) {
|
|
|
|
dev_warn(mmc_dev(host),
|
|
|
|
"missing suspend/resume ops for non-removable SDIO card\n");
|
|
|
|
/* Don't remove a non-removable card - we can't re-detect it. */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Remove the SDIO card and let it be re-detected later on. */
|
|
|
|
mmc_sdio_remove(host);
|
|
|
|
mmc_claim_host(host);
|
|
|
|
mmc_detach_bus(host);
|
|
|
|
mmc_power_off(host);
|
|
|
|
mmc_release_host(host);
|
|
|
|
host->pm_flags = 0;
|
|
|
|
|
|
|
|
return 0;
|
2013-06-10 15:03:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* SDIO suspend. Suspend all functions separately.
|
|
|
|
*/
|
|
|
|
static int mmc_sdio_suspend(struct mmc_host *host)
|
|
|
|
{
|
2019-09-08 10:12:32 +00:00
|
|
|
WARN_ON(host->sdio_irqs && !mmc_card_keep_power(host));
|
|
|
|
|
2019-06-18 12:05:17 +00:00
|
|
|
/* Prevent processing of SDIO IRQs in suspended state. */
|
|
|
|
mmc_card_set_suspended(host->card);
|
|
|
|
cancel_delayed_work_sync(&host->sdio_irq_work);
|
|
|
|
|
2015-09-11 12:41:55 +00:00
|
|
|
mmc_claim_host(host);
|
|
|
|
|
|
|
|
if (mmc_card_keep_power(host) && mmc_card_wake_sdio_irq(host))
|
2020-06-09 08:14:31 +00:00
|
|
|
sdio_disable_4bit_bus(host->card);
|
2010-03-05 21:43:34 +00:00
|
|
|
|
2015-05-07 10:10:13 +00:00
|
|
|
if (!mmc_card_keep_power(host)) {
|
2013-06-10 15:03:38 +00:00
|
|
|
mmc_power_off(host);
|
2015-05-07 10:10:13 +00:00
|
|
|
} else if (host->retune_period) {
|
|
|
|
mmc_retune_timer_stop(host);
|
|
|
|
mmc_retune_needed(host);
|
|
|
|
}
|
2013-06-10 15:03:38 +00:00
|
|
|
|
2015-09-11 12:41:55 +00:00
|
|
|
mmc_release_host(host);
|
|
|
|
|
2014-02-28 11:49:00 +00:00
|
|
|
return 0;
|
2009-09-22 23:45:28 +00:00
|
|
|
}
|
|
|
|
|
2009-09-22 23:45:29 +00:00
|
|
|
static int mmc_sdio_resume(struct mmc_host *host)
|
2009-09-22 23:45:28 +00:00
|
|
|
{
|
2014-02-28 11:49:00 +00:00
|
|
|
int err = 0;
|
2009-09-22 23:45:28 +00:00
|
|
|
|
2009-09-22 23:45:29 +00:00
|
|
|
/* Basic card reinitialization. */
|
2009-09-22 23:45:28 +00:00
|
|
|
mmc_claim_host(host);
|
2010-11-28 05:21:29 +00:00
|
|
|
|
2019-06-17 22:52:59 +00:00
|
|
|
/*
|
|
|
|
* Restore power and reinitialize the card when needed. Note that a
|
|
|
|
* removable card is checked from a detect work later on in the resume
|
|
|
|
* process.
|
|
|
|
*/
|
2013-06-10 15:03:38 +00:00
|
|
|
if (!mmc_card_keep_power(host)) {
|
2013-09-13 09:31:33 +00:00
|
|
|
mmc_power_up(host, host->card->ocr);
|
2013-06-10 15:03:38 +00:00
|
|
|
/*
|
|
|
|
* Tell runtime PM core we just powered up the card,
|
|
|
|
* since it still believes the card is powered off.
|
|
|
|
* Note that currently runtime PM is only enabled
|
|
|
|
* for SDIO cards that are MMC_CAP_POWER_OFF_CARD
|
|
|
|
*/
|
|
|
|
if (host->caps & MMC_CAP_POWER_OFF_CARD) {
|
|
|
|
pm_runtime_disable(&host->card->dev);
|
|
|
|
pm_runtime_set_active(&host->card->dev);
|
|
|
|
pm_runtime_enable(&host->card->dev);
|
|
|
|
}
|
2019-06-17 23:13:02 +00:00
|
|
|
err = mmc_sdio_reinit_card(host);
|
2019-06-17 22:52:59 +00:00
|
|
|
} else if (mmc_card_wake_sdio_irq(host)) {
|
2010-11-28 05:21:29 +00:00
|
|
|
/* We may have switched to 1-bit mode during suspend */
|
|
|
|
err = sdio_enable_4bit_bus(host->card);
|
|
|
|
}
|
|
|
|
|
2019-06-18 12:05:17 +00:00
|
|
|
if (err)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
/* Allow SDIO IRQs to be processed again. */
|
|
|
|
mmc_card_clr_suspended(host->card);
|
|
|
|
|
|
|
|
if (host->sdio_irqs) {
|
2015-10-02 08:56:11 +00:00
|
|
|
if (!(host->caps2 & MMC_CAP2_SDIO_IRQ_NOTHREAD))
|
2014-08-18 02:48:14 +00:00
|
|
|
wake_up_process(host->sdio_irq_thread);
|
2015-10-02 08:56:11 +00:00
|
|
|
else if (host->caps & MMC_CAP_SDIO_IRQ)
|
mmc: core: Fixup processing of SDIO IRQs during system suspend/resume
System suspend/resume of SDIO cards, with SDIO IRQs enabled and when using
MMC_CAP2_SDIO_IRQ_NOTHREAD is unfortunate still suffering from a fragile
behaviour. Some problems have been taken care of so far, but more issues
remains.
For example, calling the ->ack_sdio_irq() callback to let host drivers
re-enable the SDIO IRQs is a bad idea, unless the IRQ have been consumed,
which may not be the case during system suspend/resume. This may lead to
that a host driver re-signals the same SDIO IRQ over and over again,
causing a storm of IRQs and gives a ping-pong effect towards the
sdio_irq_work().
Moreover, calling the ->enable_sdio_irq() callback at system resume to
re-enable already enabled SDIO IRQs for the host, causes the runtime PM
count for some host drivers to become in-balanced. This then leads to the
host to remain runtime resumed, no matter if it's needed or not.
To fix these problems, let's check if process_sdio_pending_irqs() actually
consumed the SDIO IRQ, before we continue to ack the IRQ by invoking the
->ack_sdio_irq() callback.
Additionally, there should be no need to re-enable SDIO IRQs as the host
driver already knows if they were enabled at system suspend, thus also
whether it needs to re-enable them at system resume. For this reason, drop
the call to ->enable_sdio_irq() during system resume.
In regards to these changes there is yet another issue, which is when there
is an SDIO IRQ being signaled by the host driver, but after the SDIO card
has been system suspended. Currently these IRQs are just thrown away, while
we should at least make sure to try to consume them when the SDIO card has
been system resumed. Fix this by queueing a sdio_irq_work() after we system
resumed the SDIO card.
Tested-by: Matthias Kaehlcke <mka@chromium.org>
Reviewed-by: Matthias Kaehlcke <mka@chromium.org>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
Reviewed-by: Douglas Anderson <dianders@chromium.org>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-09-08 10:12:33 +00:00
|
|
|
queue_delayed_work(system_wq, &host->sdio_irq_work, 0);
|
2014-08-18 02:48:14 +00:00
|
|
|
}
|
|
|
|
|
2019-06-18 12:05:17 +00:00
|
|
|
out:
|
2009-09-22 23:45:28 +00:00
|
|
|
mmc_release_host(host);
|
|
|
|
|
2013-06-10 15:03:38 +00:00
|
|
|
host->pm_flags &= ~MMC_PM_KEEP_POWER;
|
2009-09-22 23:45:29 +00:00
|
|
|
return err;
|
2009-09-22 23:45:28 +00:00
|
|
|
}
|
2007-05-21 18:23:20 +00:00
|
|
|
|
2013-05-02 12:02:37 +00:00
|
|
|
static int mmc_sdio_runtime_suspend(struct mmc_host *host)
|
|
|
|
{
|
|
|
|
/* No references to the card, cut the power to it. */
|
2015-09-11 12:41:55 +00:00
|
|
|
mmc_claim_host(host);
|
2013-05-02 12:02:37 +00:00
|
|
|
mmc_power_off(host);
|
2015-09-11 12:41:55 +00:00
|
|
|
mmc_release_host(host);
|
|
|
|
|
2013-05-02 12:02:37 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mmc_sdio_runtime_resume(struct mmc_host *host)
|
|
|
|
{
|
2015-09-11 12:41:55 +00:00
|
|
|
int ret;
|
|
|
|
|
2013-05-02 12:02:37 +00:00
|
|
|
/* Restore power and re-initialize. */
|
2015-09-11 12:41:55 +00:00
|
|
|
mmc_claim_host(host);
|
2013-09-13 09:31:33 +00:00
|
|
|
mmc_power_up(host, host->card->ocr);
|
2019-06-17 23:13:02 +00:00
|
|
|
ret = mmc_sdio_reinit_card(host);
|
2015-09-11 12:41:55 +00:00
|
|
|
mmc_release_host(host);
|
|
|
|
|
|
|
|
return ret;
|
2013-05-02 12:02:37 +00:00
|
|
|
}
|
|
|
|
|
mmc: core: Re-work HW reset for SDIO cards
It have turned out that it's not a good idea to unconditionally do a power
cycle and then to re-initialize the SDIO card, as currently done through
mmc_hw_reset() -> mmc_sdio_hw_reset(). This because there may be multiple
SDIO func drivers probed, who also shares the same SDIO card.
To address these scenarios, one may be tempted to use a notification
mechanism, as to allow the core to inform each of the probed func drivers,
about an ongoing HW reset. However, supporting such an operation from the
func driver point of view, may not be entirely trivial.
Therefore, let's use a more simplistic approach to solve the problem, by
instead forcing the card to be removed and re-detected, via scheduling a
rescan-work. In this way, we can rely on existing infrastructure, as the
func driver's ->remove() and ->probe() callbacks, becomes invoked to deal
with the cleanup and the re-initialization.
This solution may be considered as rather heavy, especially if a func
driver doesn't share its card with other func drivers. To address this,
let's keep the current immediate HW reset option as well, but run it only
when there is one func driver probed for the card.
Finally, to allow the caller of mmc_hw_reset(), to understand if the reset
is being asynchronously managed from a scheduled work, it returns 1
(propagated from mmc_sdio_hw_reset()). If the HW reset is executed
successfully and synchronously it returns 0, which maintains the existing
behaviour.
Reviewed-by: Douglas Anderson <dianders@chromium.org>
Tested-by: Douglas Anderson <dianders@chromium.org>
Cc: stable@vger.kernel.org # v5.4+
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-17 13:25:36 +00:00
|
|
|
/*
|
|
|
|
* SDIO HW reset
|
|
|
|
*
|
|
|
|
* Returns 0 if the HW reset was executed synchronously, returns 1 if the HW
|
|
|
|
* reset was asynchronously scheduled, else a negative error code.
|
|
|
|
*/
|
2018-04-05 11:24:43 +00:00
|
|
|
static int mmc_sdio_hw_reset(struct mmc_host *host)
|
2015-04-23 11:15:42 +00:00
|
|
|
{
|
mmc: core: Re-work HW reset for SDIO cards
It have turned out that it's not a good idea to unconditionally do a power
cycle and then to re-initialize the SDIO card, as currently done through
mmc_hw_reset() -> mmc_sdio_hw_reset(). This because there may be multiple
SDIO func drivers probed, who also shares the same SDIO card.
To address these scenarios, one may be tempted to use a notification
mechanism, as to allow the core to inform each of the probed func drivers,
about an ongoing HW reset. However, supporting such an operation from the
func driver point of view, may not be entirely trivial.
Therefore, let's use a more simplistic approach to solve the problem, by
instead forcing the card to be removed and re-detected, via scheduling a
rescan-work. In this way, we can rely on existing infrastructure, as the
func driver's ->remove() and ->probe() callbacks, becomes invoked to deal
with the cleanup and the re-initialization.
This solution may be considered as rather heavy, especially if a func
driver doesn't share its card with other func drivers. To address this,
let's keep the current immediate HW reset option as well, but run it only
when there is one func driver probed for the card.
Finally, to allow the caller of mmc_hw_reset(), to understand if the reset
is being asynchronously managed from a scheduled work, it returns 1
(propagated from mmc_sdio_hw_reset()). If the HW reset is executed
successfully and synchronously it returns 0, which maintains the existing
behaviour.
Reviewed-by: Douglas Anderson <dianders@chromium.org>
Tested-by: Douglas Anderson <dianders@chromium.org>
Cc: stable@vger.kernel.org # v5.4+
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-17 13:25:36 +00:00
|
|
|
struct mmc_card *card = host->card;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* In case the card is shared among multiple func drivers, reset the
|
|
|
|
* card through a rescan work. In this way it will be removed and
|
|
|
|
* re-detected, thus all func drivers becomes informed about it.
|
|
|
|
*/
|
|
|
|
if (atomic_read(&card->sdio_funcs_probed) > 1) {
|
|
|
|
if (mmc_card_removed(card))
|
|
|
|
return 1;
|
|
|
|
host->rescan_entered = 0;
|
|
|
|
mmc_card_set_removed(card);
|
|
|
|
_mmc_detect_change(host, 0, false);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* A single func driver has been probed, then let's skip the heavy
|
|
|
|
* hotplug dance above and execute the reset immediately.
|
|
|
|
*/
|
|
|
|
mmc_power_cycle(host, card->ocr);
|
2019-06-17 23:13:02 +00:00
|
|
|
return mmc_sdio_reinit_card(host);
|
2015-04-23 11:15:42 +00:00
|
|
|
}
|
|
|
|
|
2018-04-05 14:31:25 +00:00
|
|
|
static int mmc_sdio_sw_reset(struct mmc_host *host)
|
|
|
|
{
|
|
|
|
mmc_set_clock(host, host->f_init);
|
|
|
|
sdio_reset(host);
|
|
|
|
mmc_go_idle(host);
|
|
|
|
|
|
|
|
mmc_set_initial_state(host);
|
|
|
|
mmc_set_initial_signal_voltage(host);
|
|
|
|
|
2019-06-17 23:13:02 +00:00
|
|
|
return mmc_sdio_reinit_card(host);
|
2018-04-05 14:31:25 +00:00
|
|
|
}
|
|
|
|
|
2007-05-21 18:23:20 +00:00
|
|
|
static const struct mmc_bus_ops mmc_sdio_ops = {
|
|
|
|
.remove = mmc_sdio_remove,
|
|
|
|
.detect = mmc_sdio_detect,
|
2013-06-10 15:03:37 +00:00
|
|
|
.pre_suspend = mmc_sdio_pre_suspend,
|
2009-09-22 23:45:28 +00:00
|
|
|
.suspend = mmc_sdio_suspend,
|
|
|
|
.resume = mmc_sdio_resume,
|
2013-05-02 12:02:37 +00:00
|
|
|
.runtime_suspend = mmc_sdio_runtime_suspend,
|
|
|
|
.runtime_resume = mmc_sdio_runtime_resume,
|
2011-11-28 14:22:00 +00:00
|
|
|
.alive = mmc_sdio_alive,
|
2018-04-05 11:24:43 +00:00
|
|
|
.hw_reset = mmc_sdio_hw_reset,
|
2018-04-05 14:31:25 +00:00
|
|
|
.sw_reset = mmc_sdio_sw_reset,
|
2007-05-21 18:23:20 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Starting point for SDIO card init.
|
|
|
|
*/
|
2011-01-03 18:36:56 +00:00
|
|
|
int mmc_attach_sdio(struct mmc_host *host)
|
2007-05-21 18:23:20 +00:00
|
|
|
{
|
2011-01-03 18:36:56 +00:00
|
|
|
int err, i, funcs;
|
2013-09-13 09:31:33 +00:00
|
|
|
u32 ocr, rocr;
|
2007-05-21 18:23:20 +00:00
|
|
|
struct mmc_card *card;
|
|
|
|
|
2007-08-09 11:23:56 +00:00
|
|
|
WARN_ON(!host->claimed);
|
2007-05-21 18:23:20 +00:00
|
|
|
|
2011-01-03 18:36:56 +00:00
|
|
|
err = mmc_send_io_op_cond(host, 0, &ocr);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
2007-05-21 18:23:20 +00:00
|
|
|
mmc_attach_bus(host, &mmc_sdio_ops);
|
2010-12-08 09:04:30 +00:00
|
|
|
if (host->ocr_avail_sdio)
|
|
|
|
host->ocr_avail = host->ocr_avail_sdio;
|
2007-05-21 18:23:20 +00:00
|
|
|
|
|
|
|
|
2013-09-13 09:31:33 +00:00
|
|
|
rocr = mmc_select_voltage(host, ocr);
|
2007-05-21 18:23:20 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Can we support the voltage(s) of the card(s)?
|
|
|
|
*/
|
2013-09-13 09:31:33 +00:00
|
|
|
if (!rocr) {
|
2007-05-21 18:23:20 +00:00
|
|
|
err = -EINVAL;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2009-09-22 23:45:28 +00:00
|
|
|
* Detect and init the card.
|
2007-05-21 18:23:20 +00:00
|
|
|
*/
|
2019-06-17 23:20:50 +00:00
|
|
|
err = mmc_sdio_init_card(host, rocr, NULL);
|
2013-09-12 14:38:56 +00:00
|
|
|
if (err)
|
|
|
|
goto err;
|
|
|
|
|
2009-09-22 23:45:28 +00:00
|
|
|
card = host->card;
|
MMC core learns about SPI
Teach the MMC/SD/SDIO core about using SPI mode.
- Use mmc_host_is_spi() so enumeration works through SPI signaling
and protocols, not just the native versions.
- Provide the SPI response type flags with each request issued,
including requests from the new lock/unlock code.
- Understand that cmd->resp[0] and mmc_get_status() results for SPI
return different values than for "native" MMC/SD protocol; this
affects resetting, checking card lock status, and some others.
- Understand that some commands act a bit differently ... notably:
* OP_COND command doesn't return the OCR
* APP_CMD status doesn't have an R1_APP_CMD analogue
Those changes required some new and updated primitives:
- Provide utilities to access two SPI-only requests, and one
request that wasn't previously needed:
* mmc_spi_read_ocr() ... SPI only
* mmc_spi_set_crc() ... SPI only (override by module parm)
* mmc_send_cid() ... for use without broadcast mode
- Updated internal routines:
* Previous mmc_send_csd() modified into mmc_send_cxd_native();
it uses native "R2" responses, which include 16 bytes of data.
* Previous mmc_send_ext_csd() becomes new mmc_send_cxd_data()
helper for command-and-data access
* Bugfix to that mmc_send_cxd_data() code: dma-to-stack is
unsafe/nonportable, so kmalloc a bounce buffer instead.
- Modified mmc_send_ext_csd() now uses mmc_send_cxd_data() helper
- Modified mmc_send_csd(), and new mmc_spi_send_cid(), routines use
those helper routines based on whether they're native or SPI
The newest categories of cards supported by the MMC stack aren't expected
to work yet with SPI: MMC or SD cards with over 4GB data, and SDIO.
All those cards support SPI mode, so eventually they should work too.
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: Pierre Ossman <drzeus@drzeus.cx>
2007-08-08 16:11:32 +00:00
|
|
|
|
2010-10-02 11:54:10 +00:00
|
|
|
/*
|
2010-11-19 07:29:09 +00:00
|
|
|
* Enable runtime PM only if supported by host+card+board
|
2010-10-02 11:54:10 +00:00
|
|
|
*/
|
2010-11-19 07:29:09 +00:00
|
|
|
if (host->caps & MMC_CAP_POWER_OFF_CARD) {
|
2017-06-09 12:27:18 +00:00
|
|
|
/*
|
|
|
|
* Do not allow runtime suspend until after SDIO function
|
|
|
|
* devices are added.
|
|
|
|
*/
|
|
|
|
pm_runtime_get_noresume(&card->dev);
|
|
|
|
|
2010-11-19 07:29:09 +00:00
|
|
|
/*
|
|
|
|
* Let runtime PM core know our card is active
|
|
|
|
*/
|
|
|
|
err = pm_runtime_set_active(&card->dev);
|
|
|
|
if (err)
|
|
|
|
goto remove;
|
2010-10-02 11:54:10 +00:00
|
|
|
|
2010-11-19 07:29:09 +00:00
|
|
|
/*
|
|
|
|
* Enable runtime PM for this card
|
|
|
|
*/
|
|
|
|
pm_runtime_enable(&card->dev);
|
|
|
|
}
|
2010-10-02 11:54:10 +00:00
|
|
|
|
2007-05-21 18:23:20 +00:00
|
|
|
/*
|
|
|
|
* The number of functions on the card is encoded inside
|
|
|
|
* the ocr.
|
|
|
|
*/
|
2009-12-17 23:27:18 +00:00
|
|
|
funcs = (ocr & 0x70000000) >> 28;
|
|
|
|
card->sdio_funcs = 0;
|
2007-07-30 16:23:53 +00:00
|
|
|
|
2007-05-26 11:48:18 +00:00
|
|
|
/*
|
|
|
|
* Initialize (but don't add) all present functions.
|
|
|
|
*/
|
2009-12-17 23:27:18 +00:00
|
|
|
for (i = 0; i < funcs; i++, card->sdio_funcs++) {
|
2007-05-26 11:48:18 +00:00
|
|
|
err = sdio_init_func(host->card, i + 1);
|
|
|
|
if (err)
|
|
|
|
goto remove;
|
2010-10-02 11:54:11 +00:00
|
|
|
|
|
|
|
/*
|
2010-11-19 07:29:09 +00:00
|
|
|
* Enable Runtime PM for this func (if supported)
|
2010-10-02 11:54:11 +00:00
|
|
|
*/
|
2010-11-19 07:29:09 +00:00
|
|
|
if (host->caps & MMC_CAP_POWER_OFF_CARD)
|
|
|
|
pm_runtime_enable(&card->sdio_func[i]->dev);
|
2007-05-26 11:48:18 +00:00
|
|
|
}
|
2007-05-21 18:23:20 +00:00
|
|
|
|
2007-05-26 11:48:18 +00:00
|
|
|
/*
|
|
|
|
* First add the card to the driver model...
|
|
|
|
*/
|
2011-01-03 18:36:56 +00:00
|
|
|
mmc_release_host(host);
|
2007-05-21 18:23:20 +00:00
|
|
|
err = mmc_add_card(host->card);
|
|
|
|
if (err)
|
2007-05-26 11:48:18 +00:00
|
|
|
goto remove_added;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* ...then the SDIO functions.
|
|
|
|
*/
|
|
|
|
for (i = 0;i < funcs;i++) {
|
|
|
|
err = sdio_add_func(host->card->sdio_func[i]);
|
|
|
|
if (err)
|
|
|
|
goto remove_added;
|
|
|
|
}
|
2007-05-21 18:23:20 +00:00
|
|
|
|
2017-06-09 12:27:18 +00:00
|
|
|
if (host->caps & MMC_CAP_POWER_OFF_CARD)
|
|
|
|
pm_runtime_put(&card->dev);
|
|
|
|
|
2011-03-03 22:40:10 +00:00
|
|
|
mmc_claim_host(host);
|
2007-05-21 18:23:20 +00:00
|
|
|
return 0;
|
|
|
|
|
2007-05-26 11:48:18 +00:00
|
|
|
|
|
|
|
remove:
|
2011-01-03 18:36:56 +00:00
|
|
|
mmc_release_host(host);
|
2017-06-09 12:27:17 +00:00
|
|
|
remove_added:
|
|
|
|
/*
|
|
|
|
* The devices are being deleted so it is not necessary to disable
|
2017-06-09 12:27:18 +00:00
|
|
|
* runtime PM. Similarly we also don't pm_runtime_put() the SDIO card
|
|
|
|
* because it needs to be active to remove any function devices that
|
|
|
|
* were probed, and after that it gets deleted.
|
2017-06-09 12:27:17 +00:00
|
|
|
*/
|
|
|
|
mmc_sdio_remove(host);
|
2011-01-03 18:36:56 +00:00
|
|
|
mmc_claim_host(host);
|
2007-05-21 18:23:20 +00:00
|
|
|
err:
|
|
|
|
mmc_detach_bus(host);
|
|
|
|
|
2011-10-11 06:14:09 +00:00
|
|
|
pr_err("%s: error %d whilst initialising SDIO card\n",
|
2007-05-21 18:23:20 +00:00
|
|
|
mmc_hostname(host), err);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|