spi: pxa2xx-pci: Enable SPI on Intel Merrifield
[deliverable/linux.git] / drivers / spi / spi-pxa2xx-pci.c
index 4fd7f9802f1b12fa1d21788f490aa185eb3cbd3e..b025eaf14f0fa674515ca06e6f40972d0a5eaf39 100644 (file)
@@ -15,6 +15,7 @@
 enum {
        PORT_CE4100,
        PORT_BYT,
+       PORT_MRFLD,
        PORT_BSW0,
        PORT_BSW1,
        PORT_BSW2,
@@ -29,8 +30,11 @@ struct pxa_spi_info {
        unsigned long max_clk_rate;
 
        /* DMA channel request parameters */
+       bool (*dma_filter)(struct dma_chan *chan, void *param);
        void *tx_param;
        void *rx_param;
+
+       int (*setup)(struct pci_dev *pdev, struct pxa_spi_info *c);
 };
 
 static struct dw_dma_slave byt_tx_param = { .dst_id = 0 };
@@ -57,6 +61,56 @@ static bool lpss_dma_filter(struct dma_chan *chan, void *param)
        return true;
 }
 
+static int lpss_spi_setup(struct pci_dev *dev, struct pxa_spi_info *c)
+{
+       struct pci_dev *dma_dev;
+
+       c->num_chipselect = 1;
+       c->max_clk_rate = 50000000;
+
+       dma_dev = pci_get_slot(dev->bus, PCI_DEVFN(PCI_SLOT(dev->devfn), 0));
+
+       if (c->tx_param) {
+               struct dw_dma_slave *slave = c->tx_param;
+
+               slave->dma_dev = &dma_dev->dev;
+               slave->m_master = 0;
+               slave->p_master = 1;
+       }
+
+       if (c->rx_param) {
+               struct dw_dma_slave *slave = c->rx_param;
+
+               slave->dma_dev = &dma_dev->dev;
+               slave->m_master = 0;
+               slave->p_master = 1;
+       }
+
+       c->dma_filter = lpss_dma_filter;
+       return 0;
+}
+
+static int mrfld_spi_setup(struct pci_dev *dev, struct pxa_spi_info *c)
+{
+       switch (PCI_FUNC(dev->devfn)) {
+       case 0:
+               c->port_id = 3;
+               c->num_chipselect = 1;
+               break;
+       case 1:
+               c->port_id = 5;
+               c->num_chipselect = 4;
+               break;
+       case 2:
+               c->port_id = 6;
+               c->num_chipselect = 1;
+               break;
+       default:
+               return -ENODEV;
+       }
+       return 0;
+}
+
 static struct pxa_spi_info spi_info_configs[] = {
        [PORT_CE4100] = {
                .type = PXA25x_SSP,
@@ -67,35 +121,36 @@ static struct pxa_spi_info spi_info_configs[] = {
        [PORT_BYT] = {
                .type = LPSS_BYT_SSP,
                .port_id = 0,
-               .num_chipselect = 1,
-               .max_clk_rate = 50000000,
+               .setup = lpss_spi_setup,
                .tx_param = &byt_tx_param,
                .rx_param = &byt_rx_param,
        },
        [PORT_BSW0] = {
                .type = LPSS_BYT_SSP,
                .port_id = 0,
-               .num_chipselect = 1,
-               .max_clk_rate = 50000000,
+               .setup = lpss_spi_setup,
                .tx_param = &bsw0_tx_param,
                .rx_param = &bsw0_rx_param,
        },
        [PORT_BSW1] = {
                .type = LPSS_BYT_SSP,
                .port_id = 1,
-               .num_chipselect = 1,
-               .max_clk_rate = 50000000,
+               .setup = lpss_spi_setup,
                .tx_param = &bsw1_tx_param,
                .rx_param = &bsw1_rx_param,
        },
        [PORT_BSW2] = {
                .type = LPSS_BYT_SSP,
                .port_id = 2,
-               .num_chipselect = 1,
-               .max_clk_rate = 50000000,
+               .setup = lpss_spi_setup,
                .tx_param = &bsw2_tx_param,
                .rx_param = &bsw2_rx_param,
        },
+       [PORT_MRFLD] = {
+               .type = PXA27x_SSP,
+               .max_clk_rate = 25000000,
+               .setup = mrfld_spi_setup,
+       },
        [PORT_QUARK_X1000] = {
                .type = QUARK_X1000_SSP,
                .port_id = -1,
@@ -105,8 +160,7 @@ static struct pxa_spi_info spi_info_configs[] = {
        [PORT_LPT] = {
                .type = LPSS_LPT_SSP,
                .port_id = 0,
-               .num_chipselect = 1,
-               .max_clk_rate = 50000000,
+               .setup = lpss_spi_setup,
                .tx_param = &lpt_tx_param,
                .rx_param = &lpt_rx_param,
        },
@@ -122,7 +176,6 @@ static int pxa2xx_spi_pci_probe(struct pci_dev *dev,
        struct ssp_device *ssp;
        struct pxa_spi_info *c;
        char buf[40];
-       struct pci_dev *dma_dev;
 
        ret = pcim_enable_device(dev);
        if (ret)
@@ -133,30 +186,15 @@ static int pxa2xx_spi_pci_probe(struct pci_dev *dev,
                return ret;
 
        c = &spi_info_configs[ent->driver_data];
-
-       memset(&spi_pdata, 0, sizeof(spi_pdata));
-       spi_pdata.num_chipselect = (c->num_chipselect > 0) ?
-                                       c->num_chipselect : dev->devfn;
-
-       dma_dev = pci_get_slot(dev->bus, PCI_DEVFN(PCI_SLOT(dev->devfn), 0));
-
-       if (c->tx_param) {
-               struct dw_dma_slave *slave = c->tx_param;
-
-               slave->dma_dev = &dma_dev->dev;
-               slave->m_master = 0;
-               slave->p_master = 1;
+       if (c->setup) {
+               ret = c->setup(dev, c);
+               if (ret)
+                       return ret;
        }
 
-       if (c->rx_param) {
-               struct dw_dma_slave *slave = c->rx_param;
-
-               slave->dma_dev = &dma_dev->dev;
-               slave->m_master = 0;
-               slave->p_master = 1;
-       }
-
-       spi_pdata.dma_filter = lpss_dma_filter;
+       memset(&spi_pdata, 0, sizeof(spi_pdata));
+       spi_pdata.num_chipselect = (c->num_chipselect > 0) ? c->num_chipselect : dev->devfn;
+       spi_pdata.dma_filter = c->dma_filter;
        spi_pdata.tx_param = c->tx_param;
        spi_pdata.rx_param = c->rx_param;
        spi_pdata.enable_dma = c->rx_param && c->tx_param;
@@ -173,8 +211,8 @@ static int pxa2xx_spi_pci_probe(struct pci_dev *dev,
        ssp->type = c->type;
 
        snprintf(buf, sizeof(buf), "pxa2xx-spi.%d", ssp->port_id);
-       ssp->clk = clk_register_fixed_rate(&dev->dev, buf , NULL,
-                                       CLK_IS_ROOT, c->max_clk_rate);
+       ssp->clk = clk_register_fixed_rate(&dev->dev, buf , NULL, 0,
+                                          c->max_clk_rate);
         if (IS_ERR(ssp->clk))
                return PTR_ERR(ssp->clk);
 
@@ -211,6 +249,7 @@ static const struct pci_device_id pxa2xx_spi_pci_devices[] = {
        { PCI_VDEVICE(INTEL, 0x2e6a), PORT_CE4100 },
        { PCI_VDEVICE(INTEL, 0x0935), PORT_QUARK_X1000 },
        { PCI_VDEVICE(INTEL, 0x0f0e), PORT_BYT },
+       { PCI_VDEVICE(INTEL, 0x1194), PORT_MRFLD },
        { PCI_VDEVICE(INTEL, 0x228e), PORT_BSW0 },
        { PCI_VDEVICE(INTEL, 0x2290), PORT_BSW1 },
        { PCI_VDEVICE(INTEL, 0x22ac), PORT_BSW2 },
This page took 0.039724 seconds and 5 git commands to generate.