spi: cadence-qspi: Add direct mode support
[oweals/u-boot.git] / drivers / spi / uniphier_spi.c
1 // 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>
10 #include <time.h>
11 #include <linux/bitfield.h>
12 #include <linux/io.h>
13 #include <spi.h>
14 #include <wait_bit.h>
15
16 DECLARE_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
67 struct 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
75 struct 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
83 static 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
95 static 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
109 static 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
135 static 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
154 static 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
198 static 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
208 static 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
276 static 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
297 static 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
361 static 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
380 static 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
392 static 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
400 static const struct udevice_id uniphier_spi_ids[] = {
401         { .compatible = "socionext,uniphier-scssi" },
402         { /* Sentinel */ }
403 };
404
405 U_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 };