This patch depends on the Grant Likely's SPI patches, so this is
for RFC only.

Also, later we'll able to remove OF_GPIO dependency.

Signed-off-by: Anton Vorontsov <[EMAIL PROTECTED]>
---
 Documentation/powerpc/booting-without-of.txt |   24 ++++
 drivers/mmc/host/Kconfig                     |    7 ++
 drivers/mmc/host/Makefile                    |    2 +-
 drivers/mmc/host/of_mmc_spi.c                |  151 ++++++++++++++++++++++++++
 4 files changed, 183 insertions(+), 1 deletions(-)
 create mode 100644 drivers/mmc/host/of_mmc_spi.c

diff --git a/Documentation/powerpc/booting-without-of.txt 
b/Documentation/powerpc/booting-without-of.txt
index 423cb2b..7d0ef80 100644
--- a/Documentation/powerpc/booting-without-of.txt
+++ b/Documentation/powerpc/booting-without-of.txt
@@ -3151,7 +3151,31 @@ platforms are moved over to use the 
flattened-device-tree model.
                        };
                };
 
+    ...) MMC-over-SPI
 
+      Required properties:
+      - #address-cells : should be 0.
+      - #size-cells : should be 0.
+      - compatible : should be "linux,mmc-spi".
+      - linux,modalias  - should be "of_mmc_spi".
+      - reg : should specify SPI address (chip-select number).
+      - max-speed : (optional) maximum frequency for this device (Hz).
+      - linux,mmc-ocr-mask : (optional) Linux-specific MMC OCR mask
+        (slot voltage).
+      - gpios : (optional) may specify GPIOs in this order: Write-Protect GPIO,
+        Card-Detect GPIO.
+
+      Example:
+
+       [EMAIL PROTECTED] {
+               compatible = "linux,mmc-spi";
+               linux,modalias = "of_mmc_spi";
+               reg = <0>;
+               max-speed = <50000000>;
+               linux,mmc-ocr-mask = <0x00200000>;
+               gpios = <&sdcsr_pio 6 0   /*  WP */
+                        &sdcsr_pio 7 1>; /* nCD */
+       };
 
 VII - Marvell Discovery mv64[345]6x System Controller chips
 ===========================================================
diff --git a/drivers/mmc/host/Kconfig b/drivers/mmc/host/Kconfig
index dead617..f468544 100644
--- a/drivers/mmc/host/Kconfig
+++ b/drivers/mmc/host/Kconfig
@@ -130,3 +130,10 @@ config MMC_SPI
 
          If unsure, or if your system has no SPI master driver, say N.
 
+config OF_MMC_SPI
+       tristate "MMC/SD over SPI OpenFirmware bindings"
+       depends on MMC_SPI && SPI_MASTER_OF && OF_GPIO
+       default y
+       help
+         Say Y here to enable OpenFirmware bindings for the MMC/SD over SPI
+         driver.
diff --git a/drivers/mmc/host/Makefile b/drivers/mmc/host/Makefile
index 3877c87..d77f880 100644
--- a/drivers/mmc/host/Makefile
+++ b/drivers/mmc/host/Makefile
@@ -17,4 +17,4 @@ obj-$(CONFIG_MMC_OMAP)                += omap.o
 obj-$(CONFIG_MMC_AT91)         += at91_mci.o
 obj-$(CONFIG_MMC_TIFM_SD)      += tifm_sd.o
 obj-$(CONFIG_MMC_SPI)          += mmc_spi.o
-
+obj-$(CONFIG_OF_MMC_SPI)       += of_mmc_spi.o
diff --git a/drivers/mmc/host/of_mmc_spi.c b/drivers/mmc/host/of_mmc_spi.c
new file mode 100644
index 0000000..b8211ce
--- /dev/null
+++ b/drivers/mmc/host/of_mmc_spi.c
@@ -0,0 +1,151 @@
+/*
+ * OpenFirmware bindings for the MMC-over-SPI driver
+ *
+ * Copyright (c) MontaVista Software, Inc. 2008.
+ *
+ * Author: Anton Vorontsov <[EMAIL PROTECTED]>
+ *
+ * This program is free software; you can redistribute  it and/or modify it
+ * under  the terms of  the GNU General  Public License as published by the
+ * Free Software Foundation;  either version 2 of the  License, or (at your
+ * option) any later version.
+ */
+
+#include <linux/init.h>
+#include <linux/kernel.h>
+#include <linux/of_platform.h>
+#include <linux/spi/spi.h>
+#include <linux/spi/spi_of.h>
+#include <linux/spi/mmc_spi.h>
+#include <linux/mmc/host.h>
+#include <linux/gpio.h>
+#include <linux/of_gpio.h>
+#include "mmc_spi.h"
+
+struct of_mmc_spi {
+       int wp_gpio;
+       int cd_gpio;
+       struct mmc_spi_platform_data mmc_pdata;
+};
+
+static int mmc_get_ro(struct device *dev)
+{
+       struct of_mmc_spi *oms = dev->archdata.of_node->data;
+
+       return gpio_get_value(oms->wp_gpio);
+}
+
+static int mmc_get_cd(struct device *dev)
+{
+       struct of_mmc_spi *oms = dev->archdata.of_node->data;
+
+       return gpio_get_value(oms->cd_gpio);
+}
+
+static int of_mmc_spi_probe(struct spi_device *spi)
+{
+       int ret = -EINVAL;
+       struct device_node *np = spi->dev.archdata.of_node;
+       struct device *dev = &spi->dev;
+       struct of_mmc_spi *oms = kzalloc(sizeof(*oms), GFP_KERNEL);
+       const u32 *ocr_mask;
+       int size;
+
+       if (!oms)
+               return -ENOMEM;
+
+       /* Somebody occupied node's data? */
+       WARN_ON(spi->dev.archdata.of_node->data);
+
+       /*
+        * mmc_spi_probe will use drvdata, so we can't use it. Use node's
+        * data instead.
+        */
+       spi->dev.archdata.of_node->data = oms;
+
+       /* We don't support interrupts yet, let's poll. */
+       oms->mmc_pdata.caps |= MMC_CAP_NEEDS_POLL;
+
+       ocr_mask = of_get_property(np, "linux,mmc-ocr-mask", &size);
+       if (ocr_mask && size >= sizeof(ocr_mask))
+               oms->mmc_pdata.ocr_mask = *ocr_mask;
+
+       oms->wp_gpio = of_get_gpio(np, 0);
+       if (gpio_is_valid(oms->wp_gpio)) {
+               ret = gpio_request(oms->wp_gpio, dev->bus_id);
+               if (ret < 0)
+                       goto err_wp_gpio;
+               oms->mmc_pdata.get_ro = &mmc_get_ro;
+       }
+
+       oms->cd_gpio = of_get_gpio(np, 1);
+       if (gpio_is_valid(oms->cd_gpio)) {
+               ret = gpio_request(oms->cd_gpio, dev->bus_id);
+               if (ret < 0)
+                       goto err_cd_gpio;
+               oms->mmc_pdata.get_cd = &mmc_get_cd;
+       }
+
+       spi->dev.platform_data = &oms->mmc_pdata;
+
+       ret = mmc_spi_probe(spi);
+       if (ret)
+               goto err_mmc_spi;
+
+       return 0;
+err_mmc_spi:
+       if (gpio_is_valid(oms->cd_gpio))
+               gpio_free(oms->cd_gpio);
+err_cd_gpio:
+       if (gpio_is_valid(oms->wp_gpio))
+               gpio_free(oms->wp_gpio);
+err_wp_gpio:
+       spi->dev.archdata.of_node->data = NULL;
+       kfree(oms);
+       return ret;
+}
+
+int __devexit of_mmc_spi_remove(struct spi_device *spi)
+{
+       struct of_mmc_spi *oms = spi->dev.archdata.of_node->data;
+       int ret;
+
+       ret = mmc_spi_remove(spi);
+       if (ret)
+               return ret;
+
+       if (gpio_is_valid(oms->cd_gpio))
+               gpio_free(oms->cd_gpio);
+       if (gpio_is_valid(oms->wp_gpio))
+               gpio_free(oms->wp_gpio);
+
+       spi->dev.archdata.of_node->data = NULL;
+       kfree(oms);
+       return 0;
+}
+
+static struct spi_driver of_mmc_spi_driver = {
+       .driver = {
+               .name = "of_mmc_spi",
+               .bus = &spi_bus_type,
+               .owner = THIS_MODULE,
+       },
+       .probe = of_mmc_spi_probe,
+       .remove = __devexit_p(of_mmc_spi_remove),
+};
+
+static int __init of_mmc_spi_init(void)
+{
+       return spi_register_driver(&of_mmc_spi_driver);
+}
+module_init(of_mmc_spi_init);
+
+static void __exit of_mmc_spi_exit(void)
+{
+       spi_unregister_driver(&of_mmc_spi_driver);
+}
+module_exit(of_mmc_spi_exit);
+
+MODULE_DESCRIPTION("OpenFirmware bindings for the MMC-over-SPI driver");
+MODULE_AUTHOR("Anton Vorontsov <[EMAIL PROTECTED]>");
+MODULE_LICENSE("GPL");
-- 
1.5.5.1
_______________________________________________
Linuxppc-dev mailing list
Linuxppc-dev@ozlabs.org
https://ozlabs.org/mailman/listinfo/linuxppc-dev

Reply via email to