I've minimally tested the platform version, but on the current mmc-next I need the extra includes below. With these changes the patch looks okay to me. Note however that this patch will conflict with Jaehoon Chung's dev_pm_ops patch, so it's probably worth fixing it to apply on top of that once it's gone into mmc-next. Thanks James diff --git a/drivers/mmc/host/dw_mmc-pci.c b/drivers/mmc/host/dw_mmc-pci.c index 5734cf7..bcfbeb2 100644 --- a/drivers/mmc/host/dw_mmc-pci.c +++ b/drivers/mmc/host/dw_mmc-pci.c @@ -11,7 +11,9 @@ #include <linux/interrupt.h> #include <linux/irq.h> +#include <linux/module.h> #include <linux/pci.h> +#include <linux/slab.h> #include <linux/mmc/host.h> #include <linux/mmc/mmc.h> #include <linux/mmc/dw_mmc.h> diff --git a/drivers/mmc/host/dw_mmc-pltfm.c b/drivers/mmc/host/dw_mmc-pltfm.c index fdf148d..0de2d03 100644 --- a/drivers/mmc/host/dw_mmc-pltfm.c +++ b/drivers/mmc/host/dw_mmc-pltfm.c @@ -11,7 +11,9 @@ */ #include <linux/interrupt.h> +#include <linux/io.h> #include <linux/irq.h> +#include <linux/module.h> #include <linux/platform_device.h> #include <linux/slab.h> #include <linux/mmc/host.h> On 12/22/2011 05:43 AM, Shashidhar Hiremath wrote: > Hi James, > Any updates on this patch ? > > On Thu, Dec 1, 2011 at 4:33 PM, James Hogan <james.hogan@xxxxxxxxxx> wrote: >> On 12/01/2011 10:51 AM, Shashidhar Hiremath wrote: >>> yes, I have compiled both the variants. >> >>> CC drivers/mmc/host/dw_mmc-pltfm.o >>> drivers/mmc/host/dw_mmc-pltfm.c: In function ‘dw_mci_pltfm_probe’: >>> drivers/mmc/host/dw_mmc-pltfm.c:48: error: implicit declaration of function ‘ioremap’ >>> drivers/mmc/host/dw_mmc-pltfm.c:48: warning: assignment makes pointer from integer without a cast >>> drivers/mmc/host/dw_mmc-pltfm.c:57: error: implicit declaration of function ‘iounmap’ >> >> Okay, possibly it's because I'm compiling on x86, and the arm headers are different. >> The following fixes it for me: >> >> diff --git a/drivers/mmc/host/dw_mmc-pltfm.c b/drivers/mmc/host/dw_mmc-pltfm.c >> index fdf148d..7af2519 100644 >> --- a/drivers/mmc/host/dw_mmc-pltfm.c >> +++ b/drivers/mmc/host/dw_mmc-pltfm.c >> @@ -11,6 +11,7 @@ >> */ >> >> #include <linux/interrupt.h> >> +#include <linux/io.h> >> #include <linux/irq.h> >> #include <linux/platform_device.h> >> #include <linux/slab.h> >> >> Cheers >> James >> >>> >>> On Thu, Dec 1, 2011 at 3:16 PM, James Hogan <james.hogan@xxxxxxxxxx> wrote: >>>> On 12/01/2011 09:41 AM, Shashidhar Hiremath wrote: >>>>> Please forgive me for repetative patches.Some mailing addresses were wrong. >>>>> So, resending the patch. >>>>> Support of PCI mode for the dw_mmc driver. This Patch adds the >>>>> support for the scenario where the Synopsys Designware IP >>>>> is present on the PCI bus. The patch adds the minimal modifications >>>>> necessary for the driver to work on PCI platform. Also added separate >>>>> files for PCI and PLATFORM modes of operation. >>>>> >>>>> Signed-off-by: Shashidhar Hiremath <shashidharh@xxxxxxxxxxxxxxx> >>>> >>>> You might like to try compiling with the platform version enabled too. >>>> >>>> Cheers >>>> James >>>> >>>>> --- >>>>> v2: >>>>> *As per Suggestions by Will Newton and James Hogan >>>>> -Reduced the number of ifdefs >>>>> v3: >>>>> *As per Suggestions by Will Newton and James Hogan >>>>> -Added separate files for PCI and PLATFORM Modes similar to SDHCI driver >>>>> v4: >>>>> *As per Suggestions by James Hogan >>>>> -Fixed Indentation Issue. >>>>> -Added Proper error Handling for probe and remove sequences. >>>>> -Modified location of some code. >>>>> -Added isr_flags to dw_mmc.h and removed the ifdef PCI from driver. >>>>> v5: >>>>> *As per Suggestions by James Hogan >>>>> -removed the redundant code. >>>>> -fixed the compilation errors. >>>>> -Modified the menuconfig Message for PLATFORM mode. >>>>> drivers/mmc/host/Kconfig | 25 ++++++ >>>>> drivers/mmc/host/Makefile | 2 + >>>>> drivers/mmc/host/dw_mmc-pci.c | 152 ++++++++++++++++++++++++++++++++++++++ >>>>> drivers/mmc/host/dw_mmc-pltfm.c | 133 +++++++++++++++++++++++++++++++++ >>>>> drivers/mmc/host/dw_mmc.c | 156 +++++++++++++------------------------- >>>>> drivers/mmc/host/dw_mmc.h | 7 ++ >>>>> include/linux/mmc/dw_mmc.h | 8 ++- >>>>> 7 files changed, 379 insertions(+), 104 deletions(-) >>>>> create mode 100644 drivers/mmc/host/dw_mmc-pci.c >>>>> create mode 100644 drivers/mmc/host/dw_mmc-pltfm.c >>>>> >>>>> diff --git a/drivers/mmc/host/Kconfig b/drivers/mmc/host/Kconfig >>>>> index 87d5067..72f48b9 100644 >>>>> --- a/drivers/mmc/host/Kconfig >>>>> +++ b/drivers/mmc/host/Kconfig >>>>> @@ -534,6 +534,31 @@ config MMC_DW_IDMAC >>>>> Designware Mobile Storage IP block. This disables the external DMA >>>>> interface. >>>>> >>>>> +config MMC_DW_PLTFM >>>>> + tristate "Synopsys Designware MCI Support as platform device" >>>>> + depends on MMC_DW >>>>> + default y >>>>> + help >>>>> + This selects the common helper functions support for Host Controller >>>>> + Interface based platform driver. Please select this option if the IP >>>>> + is present as a platform device. This is the common interface for the >>>>> + Synopsys Designware IP. >>>>> + >>>>> + If you have a controller with this interface, say Y or M here. >>>>> + >>>>> + If unsure, say Y. >>>>> + >>>>> +config MMC_DW_PCI >>>>> + tristate "Synopsys Designware MCI support on PCI bus" >>>>> + depends on MMC_DW && PCI >>>>> + help >>>>> + This selects the PCI bus for the Synopsys Designware Mobile Storage IP. >>>>> + Select this option if the IP is present on PCI platform. >>>>> + >>>>> + If you have a controller with this interface, say Y or M here. >>>>> + >>>>> + If unsure, say N. >>>>> + >>>>> config MMC_SH_MMCIF >>>>> tristate "SuperH Internal MMCIF support" >>>>> depends on MMC_BLOCK && (SUPERH || ARCH_SHMOBILE) >>>>> diff --git a/drivers/mmc/host/Makefile b/drivers/mmc/host/Makefile >>>>> index b4b83f3..3aa2fa3 100644 >>>>> --- a/drivers/mmc/host/Makefile >>>>> +++ b/drivers/mmc/host/Makefile >>>>> @@ -38,6 +38,8 @@ obj-$(CONFIG_MMC_CB710) += cb710-mmc.o >>>>> obj-$(CONFIG_MMC_VIA_SDMMC) += via-sdmmc.o >>>>> obj-$(CONFIG_SDH_BFIN) += bfin_sdh.o >>>>> obj-$(CONFIG_MMC_DW) += dw_mmc.o >>>>> +obj-$(CONFIG_MMC_DW_PLTFM) += dw_mmc-pltfm.o >>>>> +obj-$(CONFIG_MMC_DW_PCI) += dw_mmc-pci.o >>>>> obj-$(CONFIG_MMC_SH_MMCIF) += sh_mmcif.o >>>>> obj-$(CONFIG_MMC_JZ4740) += jz4740_mmc.o >>>>> obj-$(CONFIG_MMC_VUB300) += vub300.o >>>>> diff --git a/drivers/mmc/host/dw_mmc-pci.c b/drivers/mmc/host/dw_mmc-pci.c >>>>> new file mode 100644 >>>>> index 0000000..5734cf7 >>>>> --- /dev/null >>>>> +++ b/drivers/mmc/host/dw_mmc-pci.c >>>>> @@ -0,0 +1,152 @@ >>>>> +/* >>>>> + * Synopsys DesignWare Multimedia Card PCI Interface driver >>>>> + * >>>>> + * Copyright (C) 2011 Vayavya Labs Pvt. Ltd. >>>>> + * >>>>> + * 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/interrupt.h> >>>>> +#include <linux/irq.h> >>>>> +#include <linux/pci.h> >>>>> +#include <linux/mmc/host.h> >>>>> +#include <linux/mmc/mmc.h> >>>>> +#include <linux/mmc/dw_mmc.h> >>>>> +#include "dw_mmc.h" >>>>> + >>>>> +#define PCI_BAR_NO 2 >>>>> +#define COMPLETE_BAR 0 >>>>> +#define DW_MCI_VENDOR_ID 0x700 >>>>> +#define DW_MCI_DEVICE_ID 0x1107 >>>>> +/* Defining the Capabilities */ >>>>> +#define DW_MCI_CAPABILITIES (MMC_CAP_4_BIT_DATA | MMC_CAP_MMC_HIGHSPEED |\ >>>>> + MMC_CAP_SD_HIGHSPEED | MMC_CAP_8_BIT_DATA |\ >>>>> + MMC_CAP_SDIO_IRQ) >>>>> + >>>>> +static struct dw_mci_board pci_board_data = { >>>>> + .num_slots = 1, >>>>> + .caps = DW_MCI_CAPABILITIES, >>>>> + .bus_hz = 33 * 1000 * 1000, >>>>> + .detect_delay_ms = 200, >>>>> + .fifo_depth = 32, >>>>> +}; >>>>> + >>>>> +static int __devinit dw_mci_pci_probe(struct pci_dev *pdev, >>>>> + const struct pci_device_id *entries) >>>>> +{ >>>>> + struct dw_mci *host; >>>>> + int ret; >>>>> + >>>>> + ret = pci_enable_device(pdev); >>>>> + if (ret) >>>>> + return ret; >>>>> + if (pci_request_regions(pdev, "dw_mmc_pci")) { >>>>> + ret = -ENODEV; >>>>> + goto err_disable_dev; >>>>> + } >>>>> + >>>>> + host = kzalloc(sizeof(struct dw_mci), GFP_KERNEL); >>>>> + if (!host) { >>>>> + ret = -ENOMEM; >>>>> + goto err_release; >>>>> + } >>>>> + >>>>> + host->irq = pdev->irq; >>>>> + host->irq_flags = IRQF_SHARED; >>>>> + host->dev = pdev->dev; >>>>> + host->pdata = &pci_board_data; >>>>> + >>>>> + host->regs = pci_iomap(pdev, PCI_BAR_NO, COMPLETE_BAR); >>>>> + if (!host->regs) { >>>>> + ret = -EIO; >>>>> + goto err_unmap; >>>>> + } >>>>> + >>>>> + pci_set_drvdata(pdev, host); >>>>> + ret = dw_mci_probe(host); >>>>> + if (ret) >>>>> + goto err_probe_failed; >>>>> + return ret; >>>>> + >>>>> +err_probe_failed: >>>>> + pci_iounmap(pdev, host->regs); >>>>> +err_unmap: >>>>> + kfree(host); >>>>> +err_release: >>>>> + pci_release_regions(pdev); >>>>> +err_disable_dev: >>>>> + pci_disable_device(pdev); >>>>> + return ret; >>>>> +} >>>>> + >>>>> +static void __devexit dw_mci_pci_remove(struct pci_dev *pdev) >>>>> +{ >>>>> + struct dw_mci *host = pci_get_drvdata(pdev); >>>>> + >>>>> + dw_mci_remove(host); >>>>> + pci_set_drvdata(pdev, NULL); >>>>> + pci_release_regions(pdev); >>>>> + pci_iounmap(pdev, host->regs); >>>>> + kfree(host); >>>>> + pci_disable_device(pdev); >>>>> +} >>>>> + >>>>> +#ifdef CONFIG_PM >>>>> +static int dw_mci_pci_suspend(struct pci_dev *pdev, pm_message_t mesg) >>>>> +{ >>>>> + int ret; >>>>> + struct dw_mci *host = pci_get_drvdata(pdev); >>>>> + >>>>> + ret = dw_mci_suspend(host); >>>>> + return ret; >>>>> +} >>>>> + >>>>> +static int dw_mci_pci_resume(struct pci_dev *pdev) >>>>> +{ >>>>> + int ret; >>>>> + struct dw_mci *host = pci_get_drvdata(pdev); >>>>> + >>>>> + ret = dw_mci_resume(host); >>>>> + return ret; >>>>> +} >>>>> + >>>>> +#else >>>>> +#define dw_mci_pci_suspend NULL >>>>> +#define dw_mci_pci_resume NULL >>>>> +#endif /* CONFIG_PM */ >>>>> + >>>>> +static DEFINE_PCI_DEVICE_TABLE(dw_mci_pci_id) = { >>>>> + { PCI_DEVICE(DW_MCI_DEVICE_ID, DW_MCI_VENDOR_ID) }, >>>>> + {} >>>>> +}; >>>>> +MODULE_DEVICE_TABLE(pci, dw_mci_pci_id); >>>>> + >>>>> +static struct pci_driver dw_mci_pci_driver = { >>>>> + .name = "dw_mmc_pci", >>>>> + .id_table = dw_mci_pci_id, >>>>> + .probe = dw_mci_pci_probe, >>>>> + .remove = dw_mci_pci_remove, >>>>> + .suspend = dw_mci_pci_suspend, >>>>> + .resume = dw_mci_pci_resume, >>>>> +}; >>>>> + >>>>> +static int __init dw_mci_init(void) >>>>> +{ >>>>> + return pci_register_driver(&dw_mci_pci_driver); >>>>> +} >>>>> + >>>>> +static void __exit dw_mci_exit(void) >>>>> +{ >>>>> + pci_unregister_driver(&dw_mci_pci_driver); >>>>> +} >>>>> + >>>>> +module_init(dw_mci_init); >>>>> +module_exit(dw_mci_exit); >>>>> + >>>>> +MODULE_DESCRIPTION("DW Multimedia Card PCI Interface driver"); >>>>> +MODULE_AUTHOR("Shashidhar Hiremath <shashidharh@xxxxxxxxxxxxxxx>"); >>>>> +MODULE_AUTHOR("VayavyaLabs Pvt. Ltd."); >>>>> +MODULE_LICENSE("GPL v2"); >>>>> diff --git a/drivers/mmc/host/dw_mmc-pltfm.c b/drivers/mmc/host/dw_mmc-pltfm.c >>>>> new file mode 100644 >>>>> index 0000000..fdf148d >>>>> --- /dev/null >>>>> +++ b/drivers/mmc/host/dw_mmc-pltfm.c >>>>> @@ -0,0 +1,133 @@ >>>>> +/* >>>>> + * Synopsys DesignWare Multimedia Card Interface driver >>>>> + * >>>>> + * Copyright (C) 2009 NXP Semiconductors >>>>> + * Copyright (C) 2009, 2010 Imagination Technologies Ltd. >>>>> + * >>>>> + * 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/interrupt.h> >>>>> +#include <linux/irq.h> >>>>> +#include <linux/platform_device.h> >>>>> +#include <linux/slab.h> >>>>> +#include <linux/mmc/host.h> >>>>> +#include <linux/mmc/mmc.h> >>>>> +#include <linux/mmc/dw_mmc.h> >>>>> +#include "dw_mmc.h" >>>>> + >>>>> +static int dw_mci_pltfm_probe(struct platform_device *pdev) >>>>> +{ >>>>> + struct dw_mci *host; >>>>> + struct resource *regs; >>>>> + int ret; >>>>> + >>>>> + host = kzalloc(sizeof(struct dw_mci), GFP_KERNEL); >>>>> + if (!host) >>>>> + return -ENOMEM; >>>>> + >>>>> + regs = platform_get_resource(pdev, IORESOURCE_MEM, 0); >>>>> + if (!regs) { >>>>> + ret = -ENXIO; >>>>> + goto err_free; >>>>> + } >>>>> + >>>>> + host->irq = platform_get_irq(pdev, 0); >>>>> + if (host->irq < 0) { >>>>> + ret = host->irq; >>>>> + goto err_free; >>>>> + } >>>>> + >>>>> + host->dev = pdev->dev; >>>>> + host->irq_flags = 0; >>>>> + host->pdata = pdev->dev.platform_data; >>>>> + ret = -ENOMEM; >>>>> + host->regs = ioremap(regs->start, resource_size(regs)); >>>>> + if (!host->regs) >>>>> + goto err_free; >>>>> + platform_set_drvdata(pdev, host); >>>>> + ret = dw_mci_probe(host); >>>>> + if (ret) >>>>> + goto err_out; >>>>> + return ret; >>>>> +err_out: >>>>> + iounmap(host->regs); >>>>> +err_free: >>>>> + kfree(host); >>>>> + return ret; >>>>> +} >>>>> + >>>>> +static int __exit dw_mci_pltfm_remove(struct platform_device *pdev) >>>>> +{ >>>>> + struct dw_mci *host = platform_get_drvdata(pdev); >>>>> + >>>>> + >>>>> + platform_set_drvdata(pdev, NULL); >>>>> + dw_mci_remove(host); >>>>> + iounmap(host->regs); >>>>> + kfree(host); >>>>> + return 0; >>>>> +} >>>>> +#ifdef CONFIG_PM >>>>> +/* >>>>> + * TODO: we should probably disable the clock to the card in the suspend path. >>>>> + */ >>>>> +static int dw_mci_pltfm_suspend(struct platform_device *pdev, pm_message_t mesg) >>>>> +{ >>>>> + >>>>> + int ret; >>>>> + struct dw_mci *host = platform_get_drvdata(pdev); >>>>> + >>>>> + ret = dw_mci_suspend(host); >>>>> + if (ret) >>>>> + return ret; >>>>> + >>>>> + return 0; >>>>> +} >>>>> + >>>>> +static int dw_mci_pltfm_resume(struct platform_device *pdev) >>>>> +{ >>>>> + >>>>> + int ret; >>>>> + struct dw_mci *host = platform_get_drvdata(pdev); >>>>> + >>>>> + ret = dw_mci_resume(host); >>>>> + if (ret) >>>>> + return ret; >>>>> + >>>>> + return 0; >>>>> +} >>>>> +#else >>>>> +#define dw_mci_pltfm_suspend NULL >>>>> +#define dw_mci_pltfm_resume NULL >>>>> +#endif /* CONFIG_PM */ >>>>> + >>>>> +static struct platform_driver dw_mci_pltfm_driver = { >>>>> + .remove = __exit_p(dw_mci_pltfm_remove), >>>>> + .suspend = dw_mci_pltfm_suspend, >>>>> + .resume = dw_mci_pltfm_resume, >>>>> + .driver = { >>>>> + .name = "dw_mmc", >>>>> + }, >>>>> +}; >>>>> + >>>>> +static int __init dw_mci_init(void) >>>>> +{ >>>>> + return platform_driver_probe(&dw_mci_pltfm_driver, dw_mci_pltfm_probe); >>>>> +} >>>>> + >>>>> +static void __exit dw_mci_exit(void) >>>>> +{ >>>>> + platform_driver_unregister(&dw_mci_pltfm_driver); >>>>> +} >>>>> + >>>>> +module_init(dw_mci_init); >>>>> +module_exit(dw_mci_exit); >>>>> + >>>>> +MODULE_DESCRIPTION("DW Multimedia Card Interface driver"); >>>>> +MODULE_AUTHOR("NXP Semiconductor VietNam"); >>>>> +MODULE_AUTHOR("Imagination Technologies Ltd"); >>>>> +MODULE_LICENSE("GPL v2"); >>>>> diff --git a/drivers/mmc/host/dw_mmc.c b/drivers/mmc/host/dw_mmc.c >>>>> index 3aaeb08..928abc0 100644 >>>>> --- a/drivers/mmc/host/dw_mmc.c >>>>> +++ b/drivers/mmc/host/dw_mmc.c >>>>> @@ -269,7 +269,7 @@ static void dw_mci_start_command(struct dw_mci *host, >>>>> struct mmc_command *cmd, u32 cmd_flags) >>>>> { >>>>> host->cmd = cmd; >>>>> - dev_vdbg(&host->pdev->dev, >>>>> + dev_vdbg(&host->dev, >>>>> "start command: ARGR=0x%08x CMDR=0x%08x\n", >>>>> cmd->arg, cmd_flags); >>>>> >>>>> @@ -302,7 +302,7 @@ static void dw_mci_dma_cleanup(struct dw_mci *host) >>>>> struct mmc_data *data = host->data; >>>>> >>>>> if (data) >>>>> - dma_unmap_sg(&host->pdev->dev, data->sg, data->sg_len, >>>>> + dma_unmap_sg(&host->dev, data->sg, data->sg_len, >>>>> ((data->flags & MMC_DATA_WRITE) >>>>> ? DMA_TO_DEVICE : DMA_FROM_DEVICE)); >>>>> } >>>>> @@ -327,7 +327,7 @@ static void dw_mci_idmac_complete_dma(struct dw_mci *host) >>>>> { >>>>> struct mmc_data *data = host->data; >>>>> >>>>> - dev_vdbg(&host->pdev->dev, "DMA complete\n"); >>>>> + dev_vdbg(&host->dev, "DMA complete\n"); >>>>> >>>>> host->dma_ops->cleanup(host); >>>>> >>>>> @@ -463,10 +463,10 @@ static int dw_mci_submit_data_dma(struct dw_mci *host, struct mmc_data *data) >>>>> else >>>>> direction = DMA_TO_DEVICE; >>>>> >>>>> - sg_len = dma_map_sg(&host->pdev->dev, data->sg, data->sg_len, >>>>> + sg_len = dma_map_sg(&host->dev, data->sg, data->sg_len, >>>>> direction); >>>>> >>>>> - dev_vdbg(&host->pdev->dev, >>>>> + dev_vdbg(&host->dev, >>>>> "sd sg_cpu: %#lx sg_dma: %#lx sg_len: %d\n", >>>>> (unsigned long)host->sg_cpu, (unsigned long)host->sg_dma, >>>>> sg_len); >>>>> @@ -804,12 +804,12 @@ static void dw_mci_request_end(struct dw_mci *host, struct mmc_request *mrq) >>>>> slot = list_entry(host->queue.next, >>>>> struct dw_mci_slot, queue_node); >>>>> list_del(&slot->queue_node); >>>>> - dev_vdbg(&host->pdev->dev, "list not empty: %s is next\n", >>>>> + dev_vdbg(&host->dev, "list not empty: %s is next\n", >>>>> mmc_hostname(slot->mmc)); >>>>> host->state = STATE_SENDING_CMD; >>>>> dw_mci_start_request(host, slot); >>>>> } else { >>>>> - dev_vdbg(&host->pdev->dev, "list empty\n"); >>>>> + dev_vdbg(&host->dev, "list empty\n"); >>>>> host->state = STATE_IDLE; >>>>> } >>>>> >>>>> @@ -941,7 +941,7 @@ static void dw_mci_tasklet_func(unsigned long priv) >>>>> data->bytes_xfered = 0; >>>>> data->error = -ETIMEDOUT; >>>>> } else { >>>>> - dev_err(&host->pdev->dev, >>>>> + dev_err(&host->dev, >>>>> "data FIFO error " >>>>> "(status=%08x)\n", >>>>> status); >>>>> @@ -1651,7 +1651,7 @@ static int __init dw_mci_init_slot(struct dw_mci *host, unsigned int id) >>>>> struct mmc_host *mmc; >>>>> struct dw_mci_slot *slot; >>>>> >>>>> - mmc = mmc_alloc_host(sizeof(struct dw_mci_slot), &host->pdev->dev); >>>>> + mmc = mmc_alloc_host(sizeof(struct dw_mci_slot), &host->dev); >>>>> if (!mmc) >>>>> return -ENOMEM; >>>>> >>>>> @@ -1757,10 +1757,10 @@ static void dw_mci_cleanup_slot(struct dw_mci_slot *slot, unsigned int id) >>>>> static void dw_mci_init_dma(struct dw_mci *host) >>>>> { >>>>> /* Alloc memory for sg translation */ >>>>> - host->sg_cpu = dma_alloc_coherent(&host->pdev->dev, PAGE_SIZE, >>>>> + host->sg_cpu = dma_alloc_coherent(&host->dev, PAGE_SIZE, >>>>> &host->sg_dma, GFP_KERNEL); >>>>> if (!host->sg_cpu) { >>>>> - dev_err(&host->pdev->dev, "%s: could not alloc DMA memory\n", >>>>> + dev_err(&host->dev, "%s: could not alloc DMA memory\n", >>>>> __func__); >>>>> goto no_dma; >>>>> } >>>>> @@ -1768,7 +1768,7 @@ static void dw_mci_init_dma(struct dw_mci *host) >>>>> /* Determine which DMA interface to use */ >>>>> #ifdef CONFIG_MMC_DW_IDMAC >>>>> host->dma_ops = &dw_mci_idmac_ops; >>>>> - dev_info(&host->pdev->dev, "Using internal DMA controller.\n"); >>>>> + dev_info(&host->dev, "Using internal DMA controller.\n"); >>>>> #endif >>>>> >>>>> if (!host->dma_ops) >>>>> @@ -1776,12 +1776,12 @@ static void dw_mci_init_dma(struct dw_mci *host) >>>>> >>>>> if (host->dma_ops->init) { >>>>> if (host->dma_ops->init(host)) { >>>>> - dev_err(&host->pdev->dev, "%s: Unable to initialize " >>>>> + dev_err(&host->dev, "%s: Unable to initialize " >>>>> "DMA Controller.\n", __func__); >>>>> goto no_dma; >>>>> } >>>>> } else { >>>>> - dev_err(&host->pdev->dev, "DMA initialization not found.\n"); >>>>> + dev_err(&host->dev, "DMA initialization not found.\n"); >>>>> goto no_dma; >>>>> } >>>>> >>>>> @@ -1789,7 +1789,7 @@ static void dw_mci_init_dma(struct dw_mci *host) >>>>> return; >>>>> >>>>> no_dma: >>>>> - dev_info(&host->pdev->dev, "Using PIO mode.\n"); >>>>> + dev_info(&host->dev, "Using PIO mode.\n"); >>>>> host->use_dma = 0; >>>>> return; >>>>> } >>>>> @@ -1815,61 +1815,37 @@ static bool mci_wait_reset(struct device *dev, struct dw_mci *host) >>>>> return false; >>>>> } >>>>> >>>>> -static int dw_mci_probe(struct platform_device *pdev) >>>>> +int dw_mci_probe(struct dw_mci *host) >>>>> { >>>>> - struct dw_mci *host; >>>>> - struct resource *regs; >>>>> - struct dw_mci_board *pdata; >>>>> - int irq, ret, i, width; >>>>> + int width, i, ret = 0; >>>>> u32 fifo_size; >>>>> >>>>> - regs = platform_get_resource(pdev, IORESOURCE_MEM, 0); >>>>> - if (!regs) >>>>> - return -ENXIO; >>>>> - >>>>> - irq = platform_get_irq(pdev, 0); >>>>> - if (irq < 0) >>>>> - return irq; >>>>> - >>>>> - host = kzalloc(sizeof(struct dw_mci), GFP_KERNEL); >>>>> - if (!host) >>>>> - return -ENOMEM; >>>>> - >>>>> - host->pdev = pdev; >>>>> - host->pdata = pdata = pdev->dev.platform_data; >>>>> - if (!pdata || !pdata->init) { >>>>> - dev_err(&pdev->dev, >>>>> + if (!host->pdata || !host->pdata->init) { >>>>> + dev_err(&host->dev, >>>>> "Platform data must supply init function\n"); >>>>> - ret = -ENODEV; >>>>> - goto err_freehost; >>>>> + return -ENODEV; >>>>> } >>>>> >>>>> - if (!pdata->select_slot && pdata->num_slots > 1) { >>>>> - dev_err(&pdev->dev, >>>>> + if (!host->pdata->select_slot && host->pdata->num_slots > 1) { >>>>> + dev_err(&host->dev, >>>>> "Platform data must supply select_slot function\n"); >>>>> - ret = -ENODEV; >>>>> - goto err_freehost; >>>>> + return -ENODEV; >>>>> } >>>>> >>>>> - if (!pdata->bus_hz) { >>>>> - dev_err(&pdev->dev, >>>>> + if (!host->pdata->bus_hz) { >>>>> + dev_err(&host->dev, >>>>> "Platform data must supply bus speed\n"); >>>>> - ret = -ENODEV; >>>>> - goto err_freehost; >>>>> + return -ENODEV; >>>>> } >>>>> >>>>> - host->bus_hz = pdata->bus_hz; >>>>> - host->quirks = pdata->quirks; >>>>> + host->bus_hz = host->pdata->bus_hz; >>>>> + host->quirks = host->pdata->quirks; >>>>> >>>>> spin_lock_init(&host->lock); >>>>> INIT_LIST_HEAD(&host->queue); >>>>> >>>>> - ret = -ENOMEM; >>>>> - host->regs = ioremap(regs->start, resource_size(regs)); >>>>> - if (!host->regs) >>>>> - goto err_freehost; >>>>> >>>>> - host->dma_ops = pdata->dma_ops; >>>>> + host->dma_ops = host->pdata->dma_ops; >>>>> dw_mci_init_dma(host); >>>>> >>>>> /* >>>>> @@ -1899,7 +1875,7 @@ static int dw_mci_probe(struct platform_device *pdev) >>>>> } >>>>> >>>>> /* Reset all blocks */ >>>>> - if (!mci_wait_reset(&pdev->dev, host)) { >>>>> + if (!mci_wait_reset(&host->dev, host)) { >>>>> ret = -ENODEV; >>>>> goto err_dmaunmap; >>>>> } >>>>> @@ -1942,13 +1918,11 @@ static int dw_mci_probe(struct platform_device *pdev) >>>>> if (!dw_mci_card_workqueue) >>>>> goto err_dmaunmap; >>>>> INIT_WORK(&host->card_work, dw_mci_work_routine_card); >>>>> - >>>>> - ret = request_irq(irq, dw_mci_interrupt, 0, "dw-mci", host); >>>>> + ret = request_irq(host->irq, dw_mci_interrupt, >>>>> + host->irq_flags, "dw-mci", host); >>>>> if (ret) >>>>> goto err_workqueue; >>>>> >>>>> - platform_set_drvdata(pdev, host); >>>>> - >>>>> if (host->pdata->num_slots) >>>>> host->num_slots = host->pdata->num_slots; >>>>> else >>>>> @@ -1968,7 +1942,7 @@ static int dw_mci_probe(struct platform_device *pdev) >>>>> * Need to check the version-id and set data-offset for DATA register. >>>>> */ >>>>> host->verid = SDMMC_GET_VERID(mci_readl(host, VERID)); >>>>> - dev_info(&pdev->dev, "Version ID is %04x\n", host->verid); >>>>> + dev_info(&host->dev, "Version ID is %04x\n", host->verid); >>>>> >>>>> if (host->verid < DW_MMC_240A) >>>>> host->data_offset = DATA_OFFSET; >>>>> @@ -1985,12 +1959,12 @@ static int dw_mci_probe(struct platform_device *pdev) >>>>> DW_MCI_ERROR_FLAGS | SDMMC_INT_CD); >>>>> mci_writel(host, CTRL, SDMMC_CTRL_INT_ENABLE); /* Enable mci interrupt */ >>>>> >>>>> - dev_info(&pdev->dev, "DW MMC controller at irq %d, " >>>>> + dev_info(&host->dev, "DW MMC controller at irq %d, " >>>>> "%d bit host data width, " >>>>> "%u deep fifo\n", >>>>> - irq, width, fifo_size); >>>>> + host->irq, width, fifo_size); >>>>> if (host->quirks & DW_MCI_QUIRK_IDMAC_DTO) >>>>> - dev_info(&pdev->dev, "Internal DMAC interrupt fix enabled.\n"); >>>>> + dev_info(&host->dev, "Internal DMAC interrupt fix enabled.\n"); >>>>> >>>>> return 0; >>>>> >>>>> @@ -2001,7 +1975,7 @@ err_init_slot: >>>>> dw_mci_cleanup_slot(host->slot[i], i); >>>>> i--; >>>>> } >>>>> - free_irq(irq, host); >>>>> + free_irq(host->irq, host); >>>>> >>>>> err_workqueue: >>>>> destroy_workqueue(dw_mci_card_workqueue); >>>>> @@ -2009,33 +1983,26 @@ err_workqueue: >>>>> err_dmaunmap: >>>>> if (host->use_dma && host->dma_ops->exit) >>>>> host->dma_ops->exit(host); >>>>> - dma_free_coherent(&host->pdev->dev, PAGE_SIZE, >>>>> + dma_free_coherent(&host->dev, PAGE_SIZE, >>>>> host->sg_cpu, host->sg_dma); >>>>> - iounmap(host->regs); >>>>> >>>>> if (host->vmmc) { >>>>> regulator_disable(host->vmmc); >>>>> regulator_put(host->vmmc); >>>>> } >>>>> - >>>>> - >>>>> -err_freehost: >>>>> - kfree(host); >>>>> return ret; >>>>> } >>>>> +EXPORT_SYMBOL(dw_mci_probe); >>>>> >>>>> -static int __exit dw_mci_remove(struct platform_device *pdev) >>>>> +void dw_mci_remove(struct dw_mci *host) >>>>> { >>>>> - struct dw_mci *host = platform_get_drvdata(pdev); >>>>> int i; >>>>> >>>>> mci_writel(host, RINTSTS, 0xFFFFFFFF); >>>>> mci_writel(host, INTMASK, 0); /* disable all mmc interrupt first */ >>>>> >>>>> - platform_set_drvdata(pdev, NULL); >>>>> - >>>>> for (i = 0; i < host->num_slots; i++) { >>>>> - dev_dbg(&pdev->dev, "remove slot %d\n", i); >>>>> + dev_dbg(&host->dev, "remove slot %d\n", i); >>>>> if (host->slot[i]) >>>>> dw_mci_cleanup_slot(host->slot[i], i); >>>>> } >>>>> @@ -2044,9 +2011,9 @@ static int __exit dw_mci_remove(struct platform_device *pdev) >>>>> mci_writel(host, CLKENA, 0); >>>>> mci_writel(host, CLKSRC, 0); >>>>> >>>>> - free_irq(platform_get_irq(pdev, 0), host); >>>>> + free_irq(host->irq, host); >>>>> destroy_workqueue(dw_mci_card_workqueue); >>>>> - dma_free_coherent(&pdev->dev, PAGE_SIZE, host->sg_cpu, host->sg_dma); >>>>> + dma_free_coherent(&host->dev, PAGE_SIZE, host->sg_cpu, host->sg_dma); >>>>> >>>>> if (host->use_dma && host->dma_ops->exit) >>>>> host->dma_ops->exit(host); >>>>> @@ -2056,20 +2023,18 @@ static int __exit dw_mci_remove(struct platform_device *pdev) >>>>> regulator_put(host->vmmc); >>>>> } >>>>> >>>>> - iounmap(host->regs); >>>>> - >>>>> - kfree(host); >>>>> - return 0; >>>>> } >>>>> +EXPORT_SYMBOL(dw_mci_remove); >>>>> + >>>>> + >>>>> >>>>> #ifdef CONFIG_PM >>>>> /* >>>>> * TODO: we should probably disable the clock to the card in the suspend path. >>>>> */ >>>>> -static int dw_mci_suspend(struct platform_device *pdev, pm_message_t mesg) >>>>> +int dw_mci_suspend(struct dw_mci *host) >>>>> { >>>>> - int i, ret; >>>>> - struct dw_mci *host = platform_get_drvdata(pdev); >>>>> + int i, ret = 0; >>>>> >>>>> for (i = 0; i < host->num_slots; i++) { >>>>> struct dw_mci_slot *slot = host->slot[i]; >>>>> @@ -2091,19 +2056,18 @@ static int dw_mci_suspend(struct platform_device *pdev, pm_message_t mesg) >>>>> >>>>> return 0; >>>>> } >>>>> +EXPORT_SYMBOL(dw_mci_suspend); >>>>> >>>>> -static int dw_mci_resume(struct platform_device *pdev) >>>>> +int dw_mci_resume(struct dw_mci *host) >>>>> { >>>>> int i, ret; >>>>> - struct dw_mci *host = platform_get_drvdata(pdev); >>>>> - >>>>> if (host->vmmc) >>>>> regulator_enable(host->vmmc); >>>>> >>>>> if (host->dma_ops->init) >>>>> host->dma_ops->init(host); >>>>> >>>>> - if (!mci_wait_reset(&pdev->dev, host)) { >>>>> + if (!mci_wait_reset(&host->dev, host)) { >>>>> ret = -ENODEV; >>>>> return ret; >>>>> } >>>>> @@ -2125,31 +2089,19 @@ static int dw_mci_resume(struct platform_device *pdev) >>>>> if (ret < 0) >>>>> return ret; >>>>> } >>>>> - >>>>> return 0; >>>>> } >>>>> -#else >>>>> -#define dw_mci_suspend NULL >>>>> -#define dw_mci_resume NULL >>>>> +EXPORT_SYMBOL(dw_mci_resume); >>>>> #endif /* CONFIG_PM */ >>>>> >>>>> -static struct platform_driver dw_mci_driver = { >>>>> - .remove = __exit_p(dw_mci_remove), >>>>> - .suspend = dw_mci_suspend, >>>>> - .resume = dw_mci_resume, >>>>> - .driver = { >>>>> - .name = "dw_mmc", >>>>> - }, >>>>> -}; >>>>> - >>>>> static int __init dw_mci_init(void) >>>>> { >>>>> - return platform_driver_probe(&dw_mci_driver, dw_mci_probe); >>>>> + printk(KERN_INFO "Synopsys Designware Multimedia Card Interface Driver"); >>>>> + return 0; >>>>> } >>>>> >>>>> static void __exit dw_mci_exit(void) >>>>> { >>>>> - platform_driver_unregister(&dw_mci_driver); >>>>> } >>>>> >>>>> module_init(dw_mci_init); >>>>> diff --git a/drivers/mmc/host/dw_mmc.h b/drivers/mmc/host/dw_mmc.h >>>>> index 72c071f..70078f6 100644 >>>>> --- a/drivers/mmc/host/dw_mmc.h >>>>> +++ b/drivers/mmc/host/dw_mmc.h >>>>> @@ -175,4 +175,11 @@ >>>>> (*(volatile u64 __force *)((dev)->regs + SDMMC_##reg) = (value)) >>>>> #endif >>>>> >>>>> +extern int dw_mci_probe(struct dw_mci *host); >>>>> +extern void dw_mci_remove(struct dw_mci *host); >>>>> +#ifdef CONFIG_PM >>>>> +extern int dw_mci_suspend(struct dw_mci *host); >>>>> +extern int dw_mci_resume(struct dw_mci *host); >>>>> +#endif >>>>> + >>>>> #endif /* _DW_MMC_H_ */ >>>>> diff --git a/include/linux/mmc/dw_mmc.h b/include/linux/mmc/dw_mmc.h >>>>> index 6dc9b80..9cf592c 100644 >>>>> --- a/include/linux/mmc/dw_mmc.h >>>>> +++ b/include/linux/mmc/dw_mmc.h >>>>> @@ -74,7 +74,7 @@ struct mmc_data; >>>>> * @num_slots: Number of slots available. >>>>> * @verid: Denote Version ID. >>>>> * @data_offset: Set the offset of DATA register according to VERID. >>>>> - * @pdev: Platform device associated with the MMC controller. >>>>> + * @dev: Device associated with the MMC controller. >>>>> * @pdata: Platform data associated with the MMC controller. >>>>> * @slot: Slots sharing this MMC controller. >>>>> * @fifo_depth: depth of FIFO. >>>>> @@ -85,6 +85,8 @@ struct mmc_data; >>>>> * @push_data: Pointer to FIFO push function. >>>>> * @pull_data: Pointer to FIFO pull function. >>>>> * @quirks: Set of quirks that apply to specific versions of the IP. >>>>> + * @irq_flags: The flags to be passed to request_irq. >>>>> + * @irq: The irq value to be passed to request_irq. >>>>> * >>>>> * Locking >>>>> * ======= >>>>> @@ -151,7 +153,7 @@ struct dw_mci { >>>>> u32 fifoth_val; >>>>> u16 verid; >>>>> u16 data_offset; >>>>> - struct platform_device *pdev; >>>>> + struct device dev; >>>>> struct dw_mci_board *pdata; >>>>> struct dw_mci_slot *slot[MAX_MCI_SLOTS]; >>>>> >>>>> @@ -172,6 +174,8 @@ struct dw_mci { >>>>> u32 quirks; >>>>> >>>>> struct regulator *vmmc; /* Power regulator */ >>>>> + unsigned long irq_flags; /* IRQ flags */ >>>>> + unsigned int irq; >>>>> }; >>>>> >>>>> /* DMA ops for Internal/External DMAC interface */ >>>> >>> >>> >>> >> > > > -- To unsubscribe from this list: send the line "unsubscribe linux-mmc" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html