]> www.pilppa.org Git - linux-2.6-omap-h63xx.git/commitdiff
MMC: OMAP: Add back cover switch support
authorJuha Yrjola Juha Yrjola <juha.yrjola@solidboot.com>
Wed, 28 Nov 2007 00:13:30 +0000 (16:13 -0800)
committerTony Lindgren <tony@atomide.com>
Wed, 28 Nov 2007 01:55:57 +0000 (17:55 -0800)
This patch adds back MMC cover switch support in a way that
supports multiple slots.

Signed-off-by: Juha Yrjola <juha.yrjola@solidboot.com>
Signed-off-by: Jarkko Lavinen <jarkko.lavinen@nokia.com>
Signed-off-by: Carlos Eduardo Aguiar <carlos.aguiar@indt.org.br>
drivers/mmc/host/omap.c

index 22d0987c0d330b579785b3fc9bde4438a6a1d8a8..d91747364178b27af410f9086f23919353275d14 100644 (file)
@@ -108,6 +108,10 @@ struct mmc_omap_slot {
        unsigned int            fclk_freq;
        unsigned                powered:1;
 
+       struct work_struct      switch_work;
+       struct timer_list       switch_timer;
+       unsigned                cover_open;
+
        struct mmc_request      *mrq;
        struct mmc_omap_host    *host;
        struct mmc_host         *mmc;
@@ -228,6 +232,25 @@ static void mmc_omap_release_slot(struct mmc_omap_slot *slot)
        spin_unlock_irqrestore(&host->slot_lock, flags);
 }
 
+static inline
+int mmc_omap_cover_is_open(struct mmc_omap_slot *slot)
+{
+       return slot->pdata->get_cover_state(mmc_dev(slot->mmc), slot->id);
+}
+
+static ssize_t
+mmc_omap_show_cover_switch(struct device *dev, struct device_attribute *attr,
+                          char *buf)
+{
+       struct mmc_host *mmc = container_of(dev, struct mmc_host, class_dev);
+       struct mmc_omap_slot *slot = mmc_priv(mmc);
+
+       return sprintf(buf, "%s\n", mmc_omap_cover_is_open(slot) ? "open" :
+                      "closed");
+}
+
+static DEVICE_ATTR(cover_switch, S_IRUGO, mmc_omap_show_cover_switch, NULL);
+
 /* Access to the R/O switch is required for production testing
  * purposes. */
 static ssize_t
@@ -564,7 +587,8 @@ static irqreturn_t mmc_omap_irq(int irq, void *dev_id)
                                                host->cmd->opcode !=
                                                MMC_SEND_OP_COND &&
                                                host->cmd->opcode !=
-                                               MMC_APP_CMD)
+                                               MMC_APP_CMD &&
+                                               !mmc_omap_cover_is_open(slot))
                                        dev_err(mmc_dev(host->mmc),
                                                "command timeout, CMD %d\n",
                                                host->cmd->opcode);
@@ -613,6 +637,42 @@ static irqreturn_t mmc_omap_irq(int irq, void *dev_id)
        return IRQ_HANDLED;
 }
 
+void omap_mmc_notify_cover_event(struct device *dev, int slot, int is_closed)
+{
+       struct mmc_omap_host *host = dev_get_drvdata(dev);
+
+       BUG_ON(slot >= host->nr_slots);
+
+       /* Other subsystems can call in here before we're initialised. */
+       if (host->nr_slots == 0 || !host->slots[slot])
+               return;
+
+       schedule_work(&host->slots[slot]->switch_work);
+}
+
+static void mmc_omap_switch_timer(unsigned long arg)
+{
+       struct mmc_omap_slot *slot = (struct mmc_omap_slot *) arg;
+
+       schedule_work(&slot->switch_work);
+}
+
+static void mmc_omap_cover_handler(struct work_struct *work)
+{
+       struct mmc_omap_slot *slot = container_of(work, struct mmc_omap_slot,
+                                                 switch_work);
+       int cover_open;
+
+       cover_open = mmc_omap_cover_is_open(slot);
+       if (cover_open != slot->cover_open) {
+               sysfs_notify(&slot->mmc->class_dev.kobj, NULL, "cover_switch");
+               slot->cover_open = cover_open;
+               dev_info(mmc_dev(slot->mmc), "cover is now %s\n",
+                        cover_open ? "open" : "closed");
+       }
+       mmc_detect_change(slot->mmc, slot->id);
+}
+
 /* Prepare to transfer the next segment of a scatterlist */
 static void
 mmc_omap_prepare_dma(struct mmc_omap_host *host, struct mmc_data *data)
@@ -1096,13 +1156,31 @@ static int __init mmc_omap_new_slot(struct mmc_omap_host *host, int id)
                        goto err_remove_host;
        }
 
+       if (slot->pdata->get_cover_state != NULL) {
+               r = device_create_file(&mmc->class_dev,
+                                       &dev_attr_cover_switch);
+               if (r < 0)
+                       goto err_remove_slot_name;
+
+               INIT_WORK(&slot->switch_work, mmc_omap_cover_handler);
+               init_timer(&slot->switch_timer);
+               slot->switch_timer.function = mmc_omap_switch_timer;
+               slot->switch_timer.data = (unsigned long) slot;
+               schedule_work(&slot->switch_work);
+       }
+
        if (slot->pdata->get_ro != NULL) {
                r = device_create_file(&mmc->class_dev,
                                        &dev_attr_ro);
+               if (r < 0)
+                       goto err_remove_cover_attr;
        }
 
        return 0;
 
+err_remove_cover_attr:
+       if (slot->pdata->get_cover_state != NULL)
+               device_remove_file(&mmc->class_dev, &dev_attr_cover_switch);
 err_remove_slot_name:
        if (slot->pdata->name != NULL)
                device_remove_file(&mmc->class_dev, &dev_attr_ro);
@@ -1117,9 +1195,14 @@ static void mmc_omap_remove_slot(struct mmc_omap_slot *slot)
 
        if (slot->pdata->name != NULL)
                device_remove_file(&mmc->class_dev, &dev_attr_slot_name);
+       if (slot->pdata->get_cover_state != NULL)
+               device_remove_file(&mmc->class_dev, &dev_attr_cover_switch);
        if (slot->pdata->get_ro != NULL)
                device_remove_file(&mmc->class_dev, &dev_attr_ro);
 
+       del_timer_sync(&slot->switch_timer);
+       flush_scheduled_work();
+
        mmc_remove_host(mmc);
        mmc_free_host(mmc);
 }