blob: 868452d8e3e11c4094527fcce17d4c5a55dc27d1 [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 *
Andy Shevchenkoe379d2c2016-07-04 12:44:27 +03004 * Copyright (C) 2016, Intel Corporation
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +01005 */
Andy Shevchenkoe379d2c2016-07-04 12:44:27 +03006#include <linux/clk-provider.h>
7#include <linux/module.h>
8#include <linux/of_device.h>
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +01009#include <linux/pci.h>
10#include <linux/platform_device.h>
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010011#include <linux/spi/pxa2xx_spi.h>
12
Mika Westerbergb729bf32014-08-19 20:29:19 +030013#include <linux/dmaengine.h>
14#include <linux/platform_data/dma-dw.h>
15
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +080016enum {
Andy Shevchenkoe379d2c2016-07-04 12:44:27 +030017 PORT_QUARK_X1000,
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +080018 PORT_BYT,
Andy Shevchenko4f470912016-07-04 12:44:25 +030019 PORT_MRFLD,
Mika Westerberg39d36532014-08-19 20:29:21 +030020 PORT_BSW0,
21 PORT_BSW1,
22 PORT_BSW2,
Andy Shevchenkoe379d2c2016-07-04 12:44:27 +030023 PORT_CE4100,
Leif Liddycaba2482016-02-20 20:20:22 +010024 PORT_LPT,
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +080025};
26
27struct pxa_spi_info {
28 enum pxa_ssp_type type;
29 int port_id;
30 int num_chipselect;
Chew, Chiau Eeafa93c92014-07-25 01:10:54 +080031 unsigned long max_clk_rate;
Mika Westerbergb729bf32014-08-19 20:29:19 +030032
33 /* DMA channel request parameters */
Andy Shevchenko743485ea2016-07-04 12:44:24 +030034 bool (*dma_filter)(struct dma_chan *chan, void *param);
Mika Westerbergb729bf32014-08-19 20:29:19 +030035 void *tx_param;
36 void *rx_param;
Andy Shevchenko743485ea2016-07-04 12:44:24 +030037
38 int (*setup)(struct pci_dev *pdev, struct pxa_spi_info *c);
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +080039};
40
Mika Westerbergb729bf32014-08-19 20:29:19 +030041static struct dw_dma_slave byt_tx_param = { .dst_id = 0 };
42static struct dw_dma_slave byt_rx_param = { .src_id = 1 };
43
Andy Shevchenko25014522017-01-02 13:47:31 +020044static struct dw_dma_slave mrfld3_tx_param = { .dst_id = 15 };
45static struct dw_dma_slave mrfld3_rx_param = { .src_id = 14 };
46static struct dw_dma_slave mrfld5_tx_param = { .dst_id = 13 };
47static struct dw_dma_slave mrfld5_rx_param = { .src_id = 12 };
48static struct dw_dma_slave mrfld6_tx_param = { .dst_id = 11 };
49static struct dw_dma_slave mrfld6_rx_param = { .src_id = 10 };
50
Mika Westerberg39d36532014-08-19 20:29:21 +030051static struct dw_dma_slave bsw0_tx_param = { .dst_id = 0 };
52static struct dw_dma_slave bsw0_rx_param = { .src_id = 1 };
53static struct dw_dma_slave bsw1_tx_param = { .dst_id = 6 };
54static struct dw_dma_slave bsw1_rx_param = { .src_id = 7 };
55static struct dw_dma_slave bsw2_tx_param = { .dst_id = 8 };
56static struct dw_dma_slave bsw2_rx_param = { .src_id = 9 };
57
Leif Liddycaba2482016-02-20 20:20:22 +010058static struct dw_dma_slave lpt_tx_param = { .dst_id = 0 };
59static struct dw_dma_slave lpt_rx_param = { .src_id = 1 };
60
Mika Westerbergb729bf32014-08-19 20:29:19 +030061static bool lpss_dma_filter(struct dma_chan *chan, void *param)
62{
63 struct dw_dma_slave *dws = param;
64
65 if (dws->dma_dev != chan->device->dev)
66 return false;
67
68 chan->private = dws;
69 return true;
70}
71
Andy Shevchenko743485ea2016-07-04 12:44:24 +030072static int lpss_spi_setup(struct pci_dev *dev, struct pxa_spi_info *c)
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010073{
Mika Westerbergb729bf32014-08-19 20:29:19 +030074 struct pci_dev *dma_dev;
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +010075
Andy Shevchenko743485ea2016-07-04 12:44:24 +030076 c->num_chipselect = 1;
77 c->max_clk_rate = 50000000;
Mika Westerbergb729bf32014-08-19 20:29:19 +030078
79 dma_dev = pci_get_slot(dev->bus, PCI_DEVFN(PCI_SLOT(dev->devfn), 0));
80
81 if (c->tx_param) {
82 struct dw_dma_slave *slave = c->tx_param;
83
84 slave->dma_dev = &dma_dev->dev;
Andy Shevchenkoc4220252016-03-18 16:24:41 +020085 slave->m_master = 0;
86 slave->p_master = 1;
Mika Westerbergb729bf32014-08-19 20:29:19 +030087 }
88
89 if (c->rx_param) {
90 struct dw_dma_slave *slave = c->rx_param;
91
92 slave->dma_dev = &dma_dev->dev;
Andy Shevchenkoc4220252016-03-18 16:24:41 +020093 slave->m_master = 0;
94 slave->p_master = 1;
Mika Westerbergb729bf32014-08-19 20:29:19 +030095 }
96
Andy Shevchenko743485ea2016-07-04 12:44:24 +030097 c->dma_filter = lpss_dma_filter;
98 return 0;
99}
100
Andy Shevchenko4f470912016-07-04 12:44:25 +0300101static int mrfld_spi_setup(struct pci_dev *dev, struct pxa_spi_info *c)
102{
Andy Shevchenko25014522017-01-02 13:47:31 +0200103 struct pci_dev *dma_dev = pci_get_slot(dev->bus, PCI_DEVFN(21, 0));
104 struct dw_dma_slave *tx, *rx;
105
Andy Shevchenko4f470912016-07-04 12:44:25 +0300106 switch (PCI_FUNC(dev->devfn)) {
107 case 0:
108 c->port_id = 3;
109 c->num_chipselect = 1;
Andy Shevchenko25014522017-01-02 13:47:31 +0200110 c->tx_param = &mrfld3_tx_param;
111 c->rx_param = &mrfld3_rx_param;
Andy Shevchenko4f470912016-07-04 12:44:25 +0300112 break;
113 case 1:
114 c->port_id = 5;
115 c->num_chipselect = 4;
Andy Shevchenko25014522017-01-02 13:47:31 +0200116 c->tx_param = &mrfld5_tx_param;
117 c->rx_param = &mrfld5_rx_param;
Andy Shevchenko4f470912016-07-04 12:44:25 +0300118 break;
119 case 2:
120 c->port_id = 6;
121 c->num_chipselect = 1;
Andy Shevchenko25014522017-01-02 13:47:31 +0200122 c->tx_param = &mrfld6_tx_param;
123 c->rx_param = &mrfld6_rx_param;
Andy Shevchenko4f470912016-07-04 12:44:25 +0300124 break;
125 default:
126 return -ENODEV;
127 }
Andy Shevchenko25014522017-01-02 13:47:31 +0200128
129 tx = c->tx_param;
130 tx->dma_dev = &dma_dev->dev;
131
132 rx = c->rx_param;
133 rx->dma_dev = &dma_dev->dev;
134
135 c->dma_filter = lpss_dma_filter;
Andy Shevchenko4f470912016-07-04 12:44:25 +0300136 return 0;
137}
138
Andy Shevchenko743485ea2016-07-04 12:44:24 +0300139static struct pxa_spi_info spi_info_configs[] = {
140 [PORT_CE4100] = {
141 .type = PXA25x_SSP,
142 .port_id = -1,
143 .num_chipselect = -1,
144 .max_clk_rate = 3686400,
145 },
146 [PORT_BYT] = {
147 .type = LPSS_BYT_SSP,
148 .port_id = 0,
149 .setup = lpss_spi_setup,
150 .tx_param = &byt_tx_param,
151 .rx_param = &byt_rx_param,
152 },
153 [PORT_BSW0] = {
Andy Shevchenkoca80ef72016-07-05 23:12:05 +0300154 .type = LPSS_BSW_SSP,
Andy Shevchenko743485ea2016-07-04 12:44:24 +0300155 .port_id = 0,
156 .setup = lpss_spi_setup,
157 .tx_param = &bsw0_tx_param,
158 .rx_param = &bsw0_rx_param,
159 },
160 [PORT_BSW1] = {
Andy Shevchenkoca80ef72016-07-05 23:12:05 +0300161 .type = LPSS_BSW_SSP,
Andy Shevchenko743485ea2016-07-04 12:44:24 +0300162 .port_id = 1,
163 .setup = lpss_spi_setup,
164 .tx_param = &bsw1_tx_param,
165 .rx_param = &bsw1_rx_param,
166 },
167 [PORT_BSW2] = {
Andy Shevchenkoca80ef72016-07-05 23:12:05 +0300168 .type = LPSS_BSW_SSP,
Andy Shevchenko743485ea2016-07-04 12:44:24 +0300169 .port_id = 2,
170 .setup = lpss_spi_setup,
171 .tx_param = &bsw2_tx_param,
172 .rx_param = &bsw2_rx_param,
173 },
Andy Shevchenko4f470912016-07-04 12:44:25 +0300174 [PORT_MRFLD] = {
175 .type = PXA27x_SSP,
176 .max_clk_rate = 25000000,
177 .setup = mrfld_spi_setup,
178 },
Andy Shevchenko743485ea2016-07-04 12:44:24 +0300179 [PORT_QUARK_X1000] = {
180 .type = QUARK_X1000_SSP,
181 .port_id = -1,
182 .num_chipselect = 1,
183 .max_clk_rate = 50000000,
184 },
185 [PORT_LPT] = {
186 .type = LPSS_LPT_SSP,
187 .port_id = 0,
188 .setup = lpss_spi_setup,
189 .tx_param = &lpt_tx_param,
190 .rx_param = &lpt_rx_param,
191 },
192};
193
194static int pxa2xx_spi_pci_probe(struct pci_dev *dev,
195 const struct pci_device_id *ent)
196{
197 struct platform_device_info pi;
198 int ret;
199 struct platform_device *pdev;
200 struct pxa2xx_spi_master spi_pdata;
201 struct ssp_device *ssp;
202 struct pxa_spi_info *c;
203 char buf[40];
204
205 ret = pcim_enable_device(dev);
206 if (ret)
207 return ret;
208
209 ret = pcim_iomap_regions(dev, 1 << 0, "PXA2xx SPI");
210 if (ret)
211 return ret;
212
213 c = &spi_info_configs[ent->driver_data];
214 if (c->setup) {
215 ret = c->setup(dev, c);
216 if (ret)
217 return ret;
218 }
219
220 memset(&spi_pdata, 0, sizeof(spi_pdata));
221 spi_pdata.num_chipselect = (c->num_chipselect > 0) ? c->num_chipselect : dev->devfn;
222 spi_pdata.dma_filter = c->dma_filter;
Mika Westerbergb729bf32014-08-19 20:29:19 +0300223 spi_pdata.tx_param = c->tx_param;
224 spi_pdata.rx_param = c->rx_param;
225 spi_pdata.enable_dma = c->rx_param && c->tx_param;
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100226
Mika Westerberg851bacf2013-01-07 12:44:33 +0200227 ssp = &spi_pdata.ssp;
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100228 ssp->phys_base = pci_resource_start(dev, 0);
Mika Westerberg02027752013-01-07 12:44:32 +0200229 ssp->mmio_base = pcim_iomap_table(dev)[0];
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100230 ssp->irq = dev->irq;
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +0800231 ssp->port_id = (c->port_id >= 0) ? c->port_id : dev->devfn;
232 ssp->type = c->type;
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100233
Chew, Chiau Eeafa93c92014-07-25 01:10:54 +0800234 snprintf(buf, sizeof(buf), "pxa2xx-spi.%d", ssp->port_id);
Stephen Boyd280af2b2016-04-19 18:10:07 -0700235 ssp->clk = clk_register_fixed_rate(&dev->dev, buf , NULL, 0,
236 c->max_clk_rate);
Chew, Chiau Eeafa93c92014-07-25 01:10:54 +0800237 if (IS_ERR(ssp->clk))
238 return PTR_ERR(ssp->clk);
239
Mika Westerberg02027752013-01-07 12:44:32 +0200240 memset(&pi, 0, sizeof(pi));
Andy Shevchenkob70cd2d2016-08-24 14:11:30 +0300241 pi.fwnode = dev->dev.fwnode;
Mika Westerberg02027752013-01-07 12:44:32 +0200242 pi.parent = &dev->dev;
243 pi.name = "pxa2xx-spi";
244 pi.id = ssp->port_id;
245 pi.data = &spi_pdata;
246 pi.size_data = sizeof(spi_pdata);
247
248 pdev = platform_device_register_full(&pi);
Chew, Chiau Eeafa93c92014-07-25 01:10:54 +0800249 if (IS_ERR(pdev)) {
250 clk_unregister(ssp->clk);
Wei Yongjund77b5382013-02-22 10:52:35 +0800251 return PTR_ERR(pdev);
Chew, Chiau Eeafa93c92014-07-25 01:10:54 +0800252 }
Mika Westerberg02027752013-01-07 12:44:32 +0200253
Mika Westerberg851bacf2013-01-07 12:44:33 +0200254 pci_set_drvdata(dev, pdev);
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100255
Mika Westerberg02027752013-01-07 12:44:32 +0200256 return 0;
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100257}
258
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +0800259static void pxa2xx_spi_pci_remove(struct pci_dev *dev)
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100260{
Mika Westerberg851bacf2013-01-07 12:44:33 +0200261 struct platform_device *pdev = pci_get_drvdata(dev);
Chew, Chiau Eeafa93c92014-07-25 01:10:54 +0800262 struct pxa2xx_spi_master *spi_pdata;
263
264 spi_pdata = dev_get_platdata(&pdev->dev);
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100265
Mika Westerberg851bacf2013-01-07 12:44:33 +0200266 platform_device_unregister(pdev);
Chew, Chiau Eeafa93c92014-07-25 01:10:54 +0800267 clk_unregister(spi_pdata->ssp.clk);
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100268}
269
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +0800270static const struct pci_device_id pxa2xx_spi_pci_devices[] = {
Weike Chene5262d02014-11-26 02:35:10 -0800271 { PCI_VDEVICE(INTEL, 0x0935), PORT_QUARK_X1000 },
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +0800272 { PCI_VDEVICE(INTEL, 0x0f0e), PORT_BYT },
Andy Shevchenko4f470912016-07-04 12:44:25 +0300273 { PCI_VDEVICE(INTEL, 0x1194), PORT_MRFLD },
Mika Westerberg39d36532014-08-19 20:29:21 +0300274 { PCI_VDEVICE(INTEL, 0x228e), PORT_BSW0 },
275 { PCI_VDEVICE(INTEL, 0x2290), PORT_BSW1 },
276 { PCI_VDEVICE(INTEL, 0x22ac), PORT_BSW2 },
Andy Shevchenkoe379d2c2016-07-04 12:44:27 +0300277 { PCI_VDEVICE(INTEL, 0x2e6a), PORT_CE4100 },
Leif Liddycaba2482016-02-20 20:20:22 +0100278 { PCI_VDEVICE(INTEL, 0x9ce6), PORT_LPT },
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100279 { },
280};
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +0800281MODULE_DEVICE_TABLE(pci, pxa2xx_spi_pci_devices);
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100282
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +0800283static struct pci_driver pxa2xx_spi_pci_driver = {
284 .name = "pxa2xx_spi_pci",
285 .id_table = pxa2xx_spi_pci_devices,
286 .probe = pxa2xx_spi_pci_probe,
287 .remove = pxa2xx_spi_pci_remove,
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100288};
289
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +0800290module_pci_driver(pxa2xx_spi_pci_driver);
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100291
Chew, Chiau Eed6ba32d2014-04-18 00:26:06 +0800292MODULE_DESCRIPTION("CE4100/LPSS PCI-SPI glue code for PXA's driver");
Sebastian Andrzej Siewiord6ea3df2010-11-24 10:17:14 +0100293MODULE_LICENSE("GPL v2");
294MODULE_AUTHOR("Sebastian Andrzej Siewior <bigeasy@linutronix.de>");