blob: c1865c92ccb94e624077e3fc69bdb3b3fff99f71 [file] [log] [blame]
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +01001/*
2 * CE4100's SPI device is more or less the same one as found on PXA
3 *
4 */
5#include <linux/pci.h>
6#include <linux/platform_device.h>
7#include <linux/of_device.h>
Paul Gortmakerd7614de2011-07-03 15:44:29 -04008#include <linux/module.h>
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +01009#include <linux/spi/pxa2xx_spi.h>
10
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +080011enum {
12 PORT_CE4100,
13 PORT_BYT,
14};
15
16struct pxa_spi_info {
17 enum pxa_ssp_type type;
18 int port_id;
19 int num_chipselect;
20 int tx_slave_id;
21 int tx_chan_id;
22 int rx_slave_id;
23 int rx_chan_id;
24};
25
26static struct pxa_spi_info spi_info_configs[] = {
27 [PORT_CE4100] = {
28 .type = PXA25x_SSP,
29 .port_id = -1,
30 .num_chipselect = -1,
31 .tx_slave_id = -1,
32 .tx_chan_id = -1,
33 .rx_slave_id = -1,
34 .rx_chan_id = -1,
35 },
36 [PORT_BYT] = {
37 .type = LPSS_SSP,
38 .port_id = 0,
39 .num_chipselect = 1,
40 .tx_slave_id = 0,
41 .tx_chan_id = 0,
42 .rx_slave_id = 1,
43 .rx_chan_id = 1,
44 },
45};
46
47static int pxa2xx_spi_pci_probe(struct pci_dev *dev,
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010048 const struct pci_device_id *ent)
49{
Mika Westerberg02027752013-01-07 12:44:32 +020050 struct platform_device_info pi;
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010051 int ret;
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010052 struct platform_device *pdev;
Sebastian Andrzej Siewior0f3e1d22011-02-03 00:31:21 +053053 struct pxa2xx_spi_master spi_pdata;
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010054 struct ssp_device *ssp;
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +080055 struct pxa_spi_info *c;
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010056
Mika Westerberg02027752013-01-07 12:44:32 +020057 ret = pcim_enable_device(dev);
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010058 if (ret)
59 return ret;
60
Mika Westerberg02027752013-01-07 12:44:32 +020061 ret = pcim_iomap_regions(dev, 1 << 0, "PXA2xx SPI");
Mika Westerbergc1346342013-03-05 12:05:16 +020062 if (ret)
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010063 return ret;
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010064
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +080065 c = &spi_info_configs[ent->driver_data];
66
Sebastian Andrzej Siewior0f3e1d22011-02-03 00:31:21 +053067 memset(&spi_pdata, 0, sizeof(spi_pdata));
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +080068 spi_pdata.num_chipselect = (c->num_chipselect > 0) ?
69 c->num_chipselect : dev->devfn;
70 spi_pdata.tx_slave_id = c->tx_slave_id;
71 spi_pdata.tx_chan_id = c->tx_chan_id;
72 spi_pdata.rx_slave_id = c->rx_slave_id;
73 spi_pdata.rx_chan_id = c->rx_chan_id;
74 spi_pdata.enable_dma = c->rx_slave_id >= 0 && c->tx_slave_id >= 0;
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010075
Mika Westerberg851bacf2013-01-07 12:44:33 +020076 ssp = &spi_pdata.ssp;
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010077 ssp->phys_base = pci_resource_start(dev, 0);
Mika Westerberg02027752013-01-07 12:44:32 +020078 ssp->mmio_base = pcim_iomap_table(dev)[0];
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010079 if (!ssp->mmio_base) {
Mika Westerberg02027752013-01-07 12:44:32 +020080 dev_err(&dev->dev, "failed to ioremap() registers\n");
81 return -EIO;
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010082 }
83 ssp->irq = dev->irq;
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +080084 ssp->port_id = (c->port_id >= 0) ? c->port_id : dev->devfn;
85 ssp->type = c->type;
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010086
Mika Westerberg02027752013-01-07 12:44:32 +020087 memset(&pi, 0, sizeof(pi));
88 pi.parent = &dev->dev;
89 pi.name = "pxa2xx-spi";
90 pi.id = ssp->port_id;
91 pi.data = &spi_pdata;
92 pi.size_data = sizeof(spi_pdata);
93
94 pdev = platform_device_register_full(&pi);
Wei Yongjund77b5382013-02-22 10:52:35 +080095 if (IS_ERR(pdev))
96 return PTR_ERR(pdev);
Mika Westerberg02027752013-01-07 12:44:32 +020097
Mika Westerberg851bacf2013-01-07 12:44:33 +020098 pci_set_drvdata(dev, pdev);
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010099
Mika Westerberg02027752013-01-07 12:44:32 +0200100 return 0;
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100101}
102
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +0800103static void pxa2xx_spi_pci_remove(struct pci_dev *dev)
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100104{
Mika Westerberg851bacf2013-01-07 12:44:33 +0200105 struct platform_device *pdev = pci_get_drvdata(dev);
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100106
Mika Westerberg851bacf2013-01-07 12:44:33 +0200107 platform_device_unregister(pdev);
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100108}
109
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +0800110static const struct pci_device_id pxa2xx_spi_pci_devices[] = {
111 { PCI_VDEVICE(INTEL, 0x2e6a), PORT_CE4100 },
112 { PCI_VDEVICE(INTEL, 0x0f0e), PORT_BYT },
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100113 { },
114};
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +0800115MODULE_DEVICE_TABLE(pci, pxa2xx_spi_pci_devices);
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100116
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +0800117static struct pci_driver pxa2xx_spi_pci_driver = {
118 .name = "pxa2xx_spi_pci",
119 .id_table = pxa2xx_spi_pci_devices,
120 .probe = pxa2xx_spi_pci_probe,
121 .remove = pxa2xx_spi_pci_remove,
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100122};
123
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +0800124module_pci_driver(pxa2xx_spi_pci_driver);
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100125
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +0800126MODULE_DESCRIPTION("CE4100/LPSS PCI-SPI glue code for PXA's driver");
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100127MODULE_LICENSE("GPL v2");
128MODULE_AUTHOR("Sebastian Andrzej Siewior <bigeasy@linutronix.de>");