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 */ >>> >> >> >> > -- regards, Shashidhar Hiremath -- 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