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