diff options
Diffstat (limited to 'drivers/mmc/core/bus.c')
-rw-r--r-- | drivers/mmc/core/bus.c | 454 |
1 files changed, 454 insertions, 0 deletions
diff --git a/drivers/mmc/core/bus.c b/drivers/mmc/core/bus.c new file mode 100644 index 00000000..e2cbbc3c --- /dev/null +++ b/drivers/mmc/core/bus.c @@ -0,0 +1,454 @@ +/* + * linux/drivers/mmc/core/bus.c + * + * Copyright (C) 2003 Russell King, All Rights Reserved. + * Copyright (C) 2007 Pierre Ossman + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * MMC card bus driver model + */ + +#include <linux/export.h> +#include <linux/device.h> +#include <linux/err.h> +#include <linux/slab.h> +#include <linux/stat.h> +#include <linux/pm_runtime.h> + +#include <linux/mmc/card.h> +#include <linux/mmc/host.h> + +#include "core.h" +#include "sdio_cis.h" +#include "bus.h" + +#define to_mmc_driver(d) container_of(d, struct mmc_driver, drv) + +#if 0 +#define DBG(x...) printk(KERN_ALERT x) +#else +#define DBG(x...) do { } while (0) +#endif + +static ssize_t mmc_type_show(struct device *dev, + struct device_attribute *attr, char *buf) +{ + struct mmc_card *card = mmc_dev_to_card(dev); + + DBG("[%s] s\n",__func__); + + switch (card->type) { + case MMC_TYPE_MMC: + DBG("[%s] e1\n",__func__); + return sprintf(buf, "MMC\n"); + case MMC_TYPE_SD: + DBG("[%s] e2\n",__func__); + return sprintf(buf, "SD\n"); + case MMC_TYPE_SDIO: + DBG("[%s] e3\n",__func__); + return sprintf(buf, "SDIO\n"); + case MMC_TYPE_SD_COMBO: + DBG("[%s] e4\n",__func__); + return sprintf(buf, "SDcombo\n"); + default: + DBG("[%s] e5\n",__func__); + return -EFAULT; + } +} + +static struct device_attribute mmc_dev_attrs[] = { + __ATTR(type, S_IRUGO, mmc_type_show, NULL), + __ATTR_NULL, +}; + +/* + * This currently matches any MMC driver to any MMC card - drivers + * themselves make the decision whether to drive this card in their + * probe method. + */ +static int mmc_bus_match(struct device *dev, struct device_driver *drv) +{ + DBG("[%s]\n",__func__); + return 1; +} + +static int +mmc_bus_uevent(struct device *dev, struct kobj_uevent_env *env) +{ + struct mmc_card *card = mmc_dev_to_card(dev); + const char *type; + int retval = 0; + + DBG("[%s] s\n",__func__); + + switch (card->type) { + case MMC_TYPE_MMC: + type = "MMC"; + break; + case MMC_TYPE_SD: + type = "SD"; + break; + case MMC_TYPE_SDIO: + type = "SDIO"; + break; + case MMC_TYPE_SD_COMBO: + type = "SDcombo"; + break; + default: + type = NULL; + } + + if (type) { + retval = add_uevent_var(env, "MMC_TYPE=%s", type); + if (retval) { + DBG("[%s] e1\n",__func__); + return retval; + } + } + + retval = add_uevent_var(env, "MMC_NAME=%s", mmc_card_name(card)); + if (retval) { + DBG("[%s] e2\n",__func__); + return retval; + } + + /* + * Request the mmc_block device. Note: that this is a direct request + * for the module it carries no information as to what is inserted. + */ + retval = add_uevent_var(env, "MODALIAS=mmc:block"); + + DBG("[%s] e3\n",__func__); + return retval; +} + +static int mmc_bus_probe(struct device *dev) +{ + struct mmc_driver *drv = to_mmc_driver(dev->driver); + struct mmc_card *card = mmc_dev_to_card(dev); + + DBG("[%s]\n",__func__); + return drv->probe(card); +} + +static int mmc_bus_remove(struct device *dev) +{ + struct mmc_driver *drv = to_mmc_driver(dev->driver); + struct mmc_card *card = mmc_dev_to_card(dev); + + DBG("[%s] s\n",__func__); + + drv->remove(card); + + DBG("[%s] e\n",__func__); + return 0; +} + +static int mmc_bus_suspend(struct device *dev) +{ + struct mmc_driver *drv = to_mmc_driver(dev->driver); + struct mmc_card *card = mmc_dev_to_card(dev); + pm_message_t state = { PM_EVENT_SUSPEND }; + int ret = 0; + + DBG("[%s] s\n",__func__); + + if (dev->driver && drv->suspend) + ret = drv->suspend(card); + + DBG("[%s] e\n",__func__); + return ret; +} + +static int mmc_bus_resume(struct device *dev) +{ + struct mmc_driver *drv = to_mmc_driver(dev->driver); + struct mmc_card *card = mmc_dev_to_card(dev); + int ret = 0; + + DBG("[%s] s\n",__func__); + + if (dev->driver && drv->resume) + ret = drv->resume(card); + + DBG("[%s] e\n",__func__); + return ret; +} + +#ifdef CONFIG_PM_RUNTIME + +static int mmc_runtime_suspend(struct device *dev) +{ + struct mmc_card *card = mmc_dev_to_card(dev); + int ret; + + DBG("[%s] s\n",__func__); + + ret = mmc_power_save_host(card->host); + + DBG("[%s] e\n",__func__); + /*return mmc_power_save_host(card->host);*/ + return ret; + +} + +static int mmc_runtime_resume(struct device *dev) +{ + struct mmc_card *card = mmc_dev_to_card(dev); + int ret; + + DBG("[%s] s\n",__func__); + + ret = mmc_power_restore_host(card->host); + + DBG("[%s] e\n",__func__); + /*return mmc_power_restore_host(card->host);*/ + return ret; + +} + +static int mmc_runtime_idle(struct device *dev) +{ + int ret; + + DBG("[%s] s\n",__func__); + + ret = pm_runtime_suspend(dev); + + DBG("[%s] e\n",__func__); + /*return pm_runtime_suspend(dev);*/ + return ret; +} + +#endif /* !CONFIG_PM_RUNTIME */ + +static const struct dev_pm_ops mmc_bus_pm_ops = { + SET_RUNTIME_PM_OPS(mmc_runtime_suspend, mmc_runtime_resume, + mmc_runtime_idle) + SET_SYSTEM_SLEEP_PM_OPS(mmc_bus_suspend, mmc_bus_resume) +}; + +static struct bus_type mmc_bus_type = { + .name = "mmc", + .dev_attrs = mmc_dev_attrs, + .match = mmc_bus_match, + .uevent = mmc_bus_uevent, + .probe = mmc_bus_probe, + .remove = mmc_bus_remove, + .pm = &mmc_bus_pm_ops, +}; + +int mmc_register_bus(void) +{ + int ret; + + DBG("[%s] s\n",__func__); + + ret = bus_register(&mmc_bus_type); + + DBG("[%s] e\n",__func__); + return ret; + /*return bus_register(&mmc_bus_type);*/ +} + +void mmc_unregister_bus(void) +{ + DBG("[%s] s\n",__func__); + + bus_unregister(&mmc_bus_type); + + DBG("[%s] e\n",__func__); +} + +/** + * mmc_register_driver - register a media driver + * @drv: MMC media driver + */ +int mmc_register_driver(struct mmc_driver *drv) +{ + int ret; + + DBG("[%s] s\n",__func__); + + drv->drv.bus = &mmc_bus_type; + ret = driver_register(&drv->drv); + + DBG("[%s] e\n",__func__); + /*return driver_register(&drv->drv);*/ + return ret; + +} + +EXPORT_SYMBOL(mmc_register_driver); + +/** + * mmc_unregister_driver - unregister a media driver + * @drv: MMC media driver + */ +void mmc_unregister_driver(struct mmc_driver *drv) +{ + DBG("[%s] s\n",__func__); + + drv->drv.bus = &mmc_bus_type; + driver_unregister(&drv->drv); + + DBG("[%s] e\n",__func__); +} + +EXPORT_SYMBOL(mmc_unregister_driver); + +static void mmc_release_card(struct device *dev) +{ + struct mmc_card *card = mmc_dev_to_card(dev); + + DBG("[%s] s\n",__func__); + + sdio_free_common_cis(card); + + if (card->info) + kfree(card->info); + + kfree(card); + + DBG("[%s] e\n",__func__); +} + +/* + * Allocate and initialise a new MMC card structure. + */ +struct mmc_card *mmc_alloc_card(struct mmc_host *host, struct device_type *type) +{ + struct mmc_card *card; + DBG("[%s] s\n",__func__); + + card = kzalloc(sizeof(struct mmc_card), GFP_KERNEL); + if (!card) { + DBG("[%s] e1\n",__func__); + return ERR_PTR(-ENOMEM); + } + + card->host = host; + + device_initialize(&card->dev); + + card->dev.parent = mmc_classdev(host); + card->dev.bus = &mmc_bus_type; + card->dev.release = mmc_release_card; + card->dev.type = type; + + DBG("[%s] e2\n",__func__); + return card; +} + +/* + * Register a new MMC card with the driver model. + */ +int mmc_add_card(struct mmc_card *card) +{ + int ret; + const char *type; + const char *uhs_bus_speed_mode = ""; + static const char *const uhs_speeds[] = { + [UHS_SDR12_BUS_SPEED] = "SDR12 ", + [UHS_SDR25_BUS_SPEED] = "SDR25 ", + [UHS_SDR50_BUS_SPEED] = "SDR50 ", + [UHS_SDR104_BUS_SPEED] = "SDR104 ", + [UHS_DDR50_BUS_SPEED] = "DDR50 ", + }; + DBG("[%s] s\n",__func__); + + dev_set_name(&card->dev, "%s:%04x", mmc_hostname(card->host), card->rca); + + switch (card->type) { + case MMC_TYPE_MMC: + type = "MMC"; + break; + case MMC_TYPE_SD: + type = "SD"; + if (mmc_card_blockaddr(card)) { + if (mmc_card_ext_capacity(card)) + type = "SDXC"; + else + type = "SDHC"; + } + break; + case MMC_TYPE_SDIO: + type = "SDIO"; + break; + case MMC_TYPE_SD_COMBO: + type = "SD-combo"; + if (mmc_card_blockaddr(card)) + type = "SDHC-combo"; + break; + default: + type = "?"; + break; + } + + if (mmc_sd_card_uhs(card) && + (card->sd_bus_speed < ARRAY_SIZE(uhs_speeds))) + uhs_bus_speed_mode = uhs_speeds[card->sd_bus_speed]; + + if (mmc_host_is_spi(card->host)) { + pr_info("%s: new %s%s%s card on SPI\n", + mmc_hostname(card->host), + mmc_card_highspeed(card) ? "high speed " : "", + mmc_card_ddr_mode(card) ? "DDR " : "", + type); + } else { + pr_info("%s: new %s%s%s%s%s card at address %04x\n", + mmc_hostname(card->host), + mmc_card_uhs(card) ? "ultra high speed " : + (mmc_card_highspeed(card) ? "high speed " : ""), + (mmc_card_hs200(card) ? "HS200 " : ""), + mmc_card_ddr_mode(card) ? "DDR " : "", + uhs_bus_speed_mode, type, card->rca); + } + +#ifdef CONFIG_DEBUG_FS + mmc_add_card_debugfs(card); +#endif + + ret = device_add(&card->dev); + if (ret) { + DBG("[%s] e1\n",__func__); + return ret; + } + + mmc_card_set_present(card); + + DBG("[%s] e2\n",__func__); + return 0; +} + +/* + * Unregister a new MMC card with the driver model, and + * (eventually) free it. + */ +void mmc_remove_card(struct mmc_card *card) +{ + DBG("[%s] s\n",__func__); + +#ifdef CONFIG_DEBUG_FS + mmc_remove_card_debugfs(card); +#endif + + if (mmc_card_present(card)) { + if (mmc_host_is_spi(card->host)) { + pr_info("%s: SPI card removed\n", + mmc_hostname(card->host)); + } else { + pr_info("%s: card %04x removed\n", + mmc_hostname(card->host), card->rca); + } + device_del(&card->dev); + } + + put_device(&card->dev); + + DBG("[%s] e\n",__func__); +} + |