blob: 8f2c0fb4b8eedae2bd1d1dcbda68c41f231ac74d [file] [log] [blame]
Kunihiko Hayashi9424ecd2019-07-05 10:03:18 +09001// SPDX-License-Identifier: GPL-2.0+
2/*
3 * uniphier_spi.c - Socionext UniPhier SPI driver
4 * Copyright 2019 Socionext, Inc.
5 */
6
7#include <clk.h>
Kunihiko Hayashi9424ecd2019-07-05 10:03:18 +09008#include <dm.h>
Simon Glassf7ae49f2020-05-10 11:40:05 -06009#include <log.h>
Simon Glass10453152019-11-14 12:57:30 -070010#include <time.h>
Simon Glass401d1c42020-10-30 21:38:53 -060011#include <asm/global_data.h>
Simon Glass336d4612020-02-03 07:36:16 -070012#include <dm/device_compat.h>
Kunihiko Hayashi9424ecd2019-07-05 10:03:18 +090013#include <linux/bitfield.h>
Simon Glasscd93d622020-05-10 11:40:13 -060014#include <linux/bitops.h>
Simon Glassc05ed002020-05-10 11:40:11 -060015#include <linux/delay.h>
Kunihiko Hayashi9424ecd2019-07-05 10:03:18 +090016#include <linux/io.h>
17#include <spi.h>
18#include <wait_bit.h>
Simon Glass1e94b462023-09-14 18:21:46 -060019#include <linux/printk.h>
Kunihiko Hayashi9424ecd2019-07-05 10:03:18 +090020
21DECLARE_GLOBAL_DATA_PTR;
22
23#define SSI_CTL 0x00
24#define SSI_CTL_EN BIT(0)
25
26#define SSI_CKS 0x04
27#define SSI_CKS_CKRAT_MASK GENMASK(7, 0)
28#define SSI_CKS_CKPHS BIT(14)
29#define SSI_CKS_CKINIT BIT(13)
30#define SSI_CKS_CKDLY BIT(12)
31
32#define SSI_TXWDS 0x08
33#define SSI_TXWDS_WDLEN_MASK GENMASK(13, 8)
34#define SSI_TXWDS_TDTF_MASK GENMASK(7, 6)
35#define SSI_TXWDS_DTLEN_MASK GENMASK(5, 0)
36
37#define SSI_RXWDS 0x0c
38#define SSI_RXWDS_RDTF_MASK GENMASK(7, 6)
39#define SSI_RXWDS_DTLEN_MASK GENMASK(5, 0)
40
41#define SSI_FPS 0x10
42#define SSI_FPS_FSPOL BIT(15)
43#define SSI_FPS_FSTRT BIT(14)
44
45#define SSI_SR 0x14
46#define SSI_SR_BUSY BIT(7)
47#define SSI_SR_TNF BIT(5)
48#define SSI_SR_RNE BIT(0)
49
50#define SSI_IE 0x18
51
52#define SSI_IC 0x1c
53#define SSI_IC_TCIC BIT(4)
54#define SSI_IC_RCIC BIT(3)
55#define SSI_IC_RORIC BIT(0)
56
57#define SSI_FC 0x20
58#define SSI_FC_TXFFL BIT(12)
59#define SSI_FC_TXFTH_MASK GENMASK(11, 8)
60#define SSI_FC_RXFFL BIT(4)
61#define SSI_FC_RXFTH_MASK GENMASK(3, 0)
62
63#define SSI_XDR 0x24 /* TXDR for write, RXDR for read */
64
65#define SSI_FIFO_DEPTH 8U
66
67#define SSI_REG_TIMEOUT (CONFIG_SYS_HZ / 100) /* 10 ms */
68#define SSI_XFER_TIMEOUT (CONFIG_SYS_HZ) /* 1 sec */
69
70#define SSI_CLK 50000000 /* internal I/O clock: 50MHz */
71
Simon Glass8a8d24b2020-12-03 16:55:23 -070072struct uniphier_spi_plat {
Kunihiko Hayashi9424ecd2019-07-05 10:03:18 +090073 void __iomem *base;
74 u32 frequency; /* input frequency */
75 u32 speed_hz;
76 uint deactivate_delay_us; /* Delay to wait after deactivate */
77 uint activate_delay_us; /* Delay to wait after activate */
78};
79
80struct uniphier_spi_priv {
81 void __iomem *base;
82 u8 mode;
83 u8 fifo_depth;
84 u8 bits_per_word;
85 ulong last_transaction_us; /* Time of last transaction end */
86};
87
88static void uniphier_spi_enable(struct uniphier_spi_priv *priv, int enable)
89{
90 u32 val;
91
92 val = readl(priv->base + SSI_CTL);
93 if (enable)
94 val |= SSI_CTL_EN;
95 else
96 val &= ~SSI_CTL_EN;
97 writel(val, priv->base + SSI_CTL);
98}
99
100static void uniphier_spi_regdump(struct uniphier_spi_priv *priv)
101{
102 pr_debug("CTL %08x\n", readl(priv->base + SSI_CTL));
103 pr_debug("CKS %08x\n", readl(priv->base + SSI_CKS));
104 pr_debug("TXWDS %08x\n", readl(priv->base + SSI_TXWDS));
105 pr_debug("RXWDS %08x\n", readl(priv->base + SSI_RXWDS));
106 pr_debug("FPS %08x\n", readl(priv->base + SSI_FPS));
107 pr_debug("SR %08x\n", readl(priv->base + SSI_SR));
108 pr_debug("IE %08x\n", readl(priv->base + SSI_IE));
109 pr_debug("IC %08x\n", readl(priv->base + SSI_IC));
110 pr_debug("FC %08x\n", readl(priv->base + SSI_FC));
111 pr_debug("XDR %08x\n", readl(priv->base + SSI_XDR));
112}
113
114static void spi_cs_activate(struct udevice *dev)
115{
116 struct udevice *bus = dev->parent;
Simon Glass0fd3d912020-12-22 19:30:28 -0700117 struct uniphier_spi_plat *plat = dev_get_plat(bus);
Kunihiko Hayashi9424ecd2019-07-05 10:03:18 +0900118 struct uniphier_spi_priv *priv = dev_get_priv(bus);
119 ulong delay_us; /* The delay completed so far */
120 u32 val;
121
122 /* If it's too soon to do another transaction, wait */
123 if (plat->deactivate_delay_us && priv->last_transaction_us) {
124 delay_us = timer_get_us() - priv->last_transaction_us;
125 if (delay_us < plat->deactivate_delay_us)
126 udelay(plat->deactivate_delay_us - delay_us);
127 }
128
129 val = readl(priv->base + SSI_FPS);
130 if (priv->mode & SPI_CS_HIGH)
131 val |= SSI_FPS_FSPOL;
132 else
133 val &= ~SSI_FPS_FSPOL;
134 writel(val, priv->base + SSI_FPS);
135
136 if (plat->activate_delay_us)
137 udelay(plat->activate_delay_us);
138}
139
140static void spi_cs_deactivate(struct udevice *dev)
141{
142 struct udevice *bus = dev->parent;
Simon Glass0fd3d912020-12-22 19:30:28 -0700143 struct uniphier_spi_plat *plat = dev_get_plat(bus);
Kunihiko Hayashi9424ecd2019-07-05 10:03:18 +0900144 struct uniphier_spi_priv *priv = dev_get_priv(bus);
145 u32 val;
146
147 val = readl(priv->base + SSI_FPS);
148 if (priv->mode & SPI_CS_HIGH)
149 val &= ~SSI_FPS_FSPOL;
150 else
151 val |= SSI_FPS_FSPOL;
152 writel(val, priv->base + SSI_FPS);
153
154 /* Remember time of this transaction so we can honour the bus delay */
155 if (plat->deactivate_delay_us)
156 priv->last_transaction_us = timer_get_us();
157}
158
159static int uniphier_spi_claim_bus(struct udevice *dev)
160{
161 struct udevice *bus = dev->parent;
162 struct uniphier_spi_priv *priv = dev_get_priv(bus);
163 u32 val, size;
164
165 uniphier_spi_enable(priv, false);
166
167 /* disable interrupts */
168 writel(0, priv->base + SSI_IE);
169
170 /* bits_per_word */
171 size = priv->bits_per_word;
172 val = readl(priv->base + SSI_TXWDS);
173 val &= ~(SSI_TXWDS_WDLEN_MASK | SSI_TXWDS_DTLEN_MASK);
174 val |= FIELD_PREP(SSI_TXWDS_WDLEN_MASK, size);
175 val |= FIELD_PREP(SSI_TXWDS_DTLEN_MASK, size);
176 writel(val, priv->base + SSI_TXWDS);
177
178 val = readl(priv->base + SSI_RXWDS);
179 val &= ~SSI_RXWDS_DTLEN_MASK;
180 val |= FIELD_PREP(SSI_RXWDS_DTLEN_MASK, size);
181 writel(val, priv->base + SSI_RXWDS);
182
183 /* reset FIFOs */
184 val = SSI_FC_TXFFL | SSI_FC_RXFFL;
185 writel(val, priv->base + SSI_FC);
186
187 /* FIFO threthold */
188 val = readl(priv->base + SSI_FC);
189 val &= ~(SSI_FC_TXFTH_MASK | SSI_FC_RXFTH_MASK);
190 val |= FIELD_PREP(SSI_FC_TXFTH_MASK, priv->fifo_depth);
191 val |= FIELD_PREP(SSI_FC_RXFTH_MASK, priv->fifo_depth);
192 writel(val, priv->base + SSI_FC);
193
194 /* clear interrupts */
195 writel(SSI_IC_TCIC | SSI_IC_RCIC | SSI_IC_RORIC,
196 priv->base + SSI_IC);
197
198 uniphier_spi_enable(priv, true);
199
200 return 0;
201}
202
203static int uniphier_spi_release_bus(struct udevice *dev)
204{
205 struct udevice *bus = dev->parent;
206 struct uniphier_spi_priv *priv = dev_get_priv(bus);
207
208 uniphier_spi_enable(priv, false);
209
210 return 0;
211}
212
213static int uniphier_spi_xfer(struct udevice *dev, unsigned int bitlen,
214 const void *dout, void *din, unsigned long flags)
215{
216 struct udevice *bus = dev->parent;
217 struct uniphier_spi_priv *priv = dev_get_priv(bus);
218 const u8 *tx_buf = dout;
219 u8 *rx_buf = din, buf;
220 u32 len = bitlen / 8;
221 u32 tx_len, rx_len;
222 u32 ts, status;
223 int ret = 0;
224
225 if (bitlen % 8) {
226 dev_err(dev, "Non byte aligned SPI transfer\n");
227 return -EINVAL;
228 }
229
230 if (flags & SPI_XFER_BEGIN)
231 spi_cs_activate(dev);
232
233 uniphier_spi_enable(priv, true);
234
235 ts = get_timer(0);
236 tx_len = len;
237 rx_len = len;
238
239 uniphier_spi_regdump(priv);
240
241 while (tx_len || rx_len) {
242 ret = wait_for_bit_le32(priv->base + SSI_SR, SSI_SR_BUSY, false,
243 SSI_REG_TIMEOUT * 1000, false);
244 if (ret) {
245 if (ret == -ETIMEDOUT)
246 dev_err(dev, "access timeout\n");
247 break;
248 }
249
250 status = readl(priv->base + SSI_SR);
251 /* write the data into TX */
252 if (tx_len && (status & SSI_SR_TNF)) {
253 buf = tx_buf ? *tx_buf++ : 0;
254 writel(buf, priv->base + SSI_XDR);
255 tx_len--;
256 }
257
258 /* read the data from RX */
259 if (rx_len && (status & SSI_SR_RNE)) {
260 buf = readl(priv->base + SSI_XDR);
261 if (rx_buf)
262 *rx_buf++ = buf;
263 rx_len--;
264 }
265
266 if (get_timer(ts) >= SSI_XFER_TIMEOUT) {
267 dev_err(dev, "transfer timeout\n");
268 ret = -ETIMEDOUT;
269 break;
270 }
271 }
272
273 if (flags & SPI_XFER_END)
274 spi_cs_deactivate(dev);
275
276 uniphier_spi_enable(priv, false);
277
278 return ret;
279}
280
281static int uniphier_spi_set_speed(struct udevice *bus, uint speed)
282{
Simon Glass0fd3d912020-12-22 19:30:28 -0700283 struct uniphier_spi_plat *plat = dev_get_plat(bus);
Kunihiko Hayashi9424ecd2019-07-05 10:03:18 +0900284 struct uniphier_spi_priv *priv = dev_get_priv(bus);
285 u32 val, ckdiv;
286
287 if (speed > plat->frequency)
288 speed = plat->frequency;
289
290 /* baudrate */
291 ckdiv = DIV_ROUND_UP(SSI_CLK, speed);
292 ckdiv = round_up(ckdiv, 2);
293
294 val = readl(priv->base + SSI_CKS);
295 val &= ~SSI_CKS_CKRAT_MASK;
296 val |= ckdiv & SSI_CKS_CKRAT_MASK;
297 writel(val, priv->base + SSI_CKS);
298
299 return 0;
300}
301
302static int uniphier_spi_set_mode(struct udevice *bus, uint mode)
303{
304 struct uniphier_spi_priv *priv = dev_get_priv(bus);
305 u32 val1, val2;
306
307 /*
308 * clock setting
309 * CKPHS capture timing. 0:rising edge, 1:falling edge
310 * CKINIT clock initial level. 0:low, 1:high
311 * CKDLY clock delay. 0:no delay, 1:delay depending on FSTRT
312 * (FSTRT=0: 1 clock, FSTRT=1: 0.5 clock)
313 *
314 * frame setting
315 * FSPOL frame signal porarity. 0: low, 1: high
316 * FSTRT start frame timing
317 * 0: rising edge of clock, 1: falling edge of clock
318 */
319 val1 = readl(priv->base + SSI_CKS);
320 val2 = readl(priv->base + SSI_FPS);
321
322 switch (mode & (SPI_CPOL | SPI_CPHA)) {
323 case SPI_MODE_0:
324 /* CKPHS=1, CKINIT=0, CKDLY=1, FSTRT=0 */
325 val1 |= SSI_CKS_CKPHS | SSI_CKS_CKDLY;
326 val1 &= ~SSI_CKS_CKINIT;
327 val2 &= ~SSI_FPS_FSTRT;
328 break;
329 case SPI_MODE_1:
330 /* CKPHS=0, CKINIT=0, CKDLY=0, FSTRT=1 */
331 val1 &= ~(SSI_CKS_CKPHS | SSI_CKS_CKINIT | SSI_CKS_CKDLY);
332 val2 |= SSI_FPS_FSTRT;
333 break;
334 case SPI_MODE_2:
335 /* CKPHS=0, CKINIT=1, CKDLY=1, FSTRT=1 */
336 val1 |= SSI_CKS_CKINIT | SSI_CKS_CKDLY;
337 val1 &= ~SSI_CKS_CKPHS;
338 val2 |= SSI_FPS_FSTRT;
339 break;
340 case SPI_MODE_3:
341 /* CKPHS=1, CKINIT=1, CKDLY=0, FSTRT=0 */
342 val1 |= SSI_CKS_CKPHS | SSI_CKS_CKINIT;
343 val1 &= ~SSI_CKS_CKDLY;
344 val2 &= ~SSI_FPS_FSTRT;
345 break;
346 }
347
348 writel(val1, priv->base + SSI_CKS);
349 writel(val2, priv->base + SSI_FPS);
350
351 /* format */
352 val1 = readl(priv->base + SSI_TXWDS);
353 val2 = readl(priv->base + SSI_RXWDS);
354 if (mode & SPI_LSB_FIRST) {
355 val1 |= FIELD_PREP(SSI_TXWDS_TDTF_MASK, 1);
356 val2 |= FIELD_PREP(SSI_RXWDS_RDTF_MASK, 1);
357 }
358 writel(val1, priv->base + SSI_TXWDS);
359 writel(val2, priv->base + SSI_RXWDS);
360
361 priv->mode = mode;
362
363 return 0;
364}
365
Simon Glassd1998a92020-12-03 16:55:21 -0700366static int uniphier_spi_of_to_plat(struct udevice *bus)
Kunihiko Hayashi9424ecd2019-07-05 10:03:18 +0900367{
Simon Glass0fd3d912020-12-22 19:30:28 -0700368 struct uniphier_spi_plat *plat = dev_get_plat(bus);
Kunihiko Hayashi9424ecd2019-07-05 10:03:18 +0900369 const void *blob = gd->fdt_blob;
370 int node = dev_of_offset(bus);
371
Masahiro Yamada702e57e2020-08-04 14:14:43 +0900372 plat->base = dev_read_addr_ptr(bus);
Kunihiko Hayashi9424ecd2019-07-05 10:03:18 +0900373
374 plat->frequency =
375 fdtdec_get_int(blob, node, "spi-max-frequency", 12500000);
376 plat->deactivate_delay_us =
377 fdtdec_get_int(blob, node, "spi-deactivate-delay", 0);
378 plat->activate_delay_us =
379 fdtdec_get_int(blob, node, "spi-activate-delay", 0);
380 plat->speed_hz = plat->frequency / 2;
381
382 return 0;
383}
384
385static int uniphier_spi_probe(struct udevice *bus)
386{
Simon Glass8a8d24b2020-12-03 16:55:23 -0700387 struct uniphier_spi_plat *plat = dev_get_plat(bus);
Kunihiko Hayashi9424ecd2019-07-05 10:03:18 +0900388 struct uniphier_spi_priv *priv = dev_get_priv(bus);
389
390 priv->base = plat->base;
391 priv->fifo_depth = SSI_FIFO_DEPTH;
392 priv->bits_per_word = 8;
393
394 return 0;
395}
396
397static const struct dm_spi_ops uniphier_spi_ops = {
398 .claim_bus = uniphier_spi_claim_bus,
399 .release_bus = uniphier_spi_release_bus,
400 .xfer = uniphier_spi_xfer,
401 .set_speed = uniphier_spi_set_speed,
402 .set_mode = uniphier_spi_set_mode,
403};
404
405static const struct udevice_id uniphier_spi_ids[] = {
406 { .compatible = "socionext,uniphier-scssi" },
407 { /* Sentinel */ }
408};
409
410U_BOOT_DRIVER(uniphier_spi) = {
411 .name = "uniphier_spi",
412 .id = UCLASS_SPI,
413 .of_match = uniphier_spi_ids,
414 .ops = &uniphier_spi_ops,
Simon Glassd1998a92020-12-03 16:55:21 -0700415 .of_to_plat = uniphier_spi_of_to_plat,
Simon Glass8a8d24b2020-12-03 16:55:23 -0700416 .plat_auto = sizeof(struct uniphier_spi_plat),
Simon Glass41575d82020-12-03 16:55:17 -0700417 .priv_auto = sizeof(struct uniphier_spi_priv),
Kunihiko Hayashi9424ecd2019-07-05 10:03:18 +0900418 .probe = uniphier_spi_probe,
419};