blob: 2125d25561adb42db439bd7d0ebe5e0169cb07e4 [file] [log] [blame]
Nathan Barrett-Morrison88720e72025-02-26 12:30:33 -05001// SPDX-License-Identifier: GPL-2.0-or-later
2/*
3 * (C) Copyright 2022 - Analog Devices, Inc.
4 *
5 * Written and/or maintained by Timesys Corporation
6 *
7 * Converted to driver model by Nathan Barrett-Morrison
8 *
9 * Contact: Nathan Barrett-Morrison <nathan.morrison@timesys.com>
10 * Contact: Greg Malysa <greg.malysa@timesys.com>
11 * Contact: Ian Roberts <ian.roberts@timesys.com>
12 * Contact: Piotr Wojtaszczyk <piotr.wojtaszczyk@timesys.com>
13 *
14 */
15
16#include <clk.h>
17#include <dm.h>
18#include <mapmem.h>
19#include <spi.h>
20#include <spi-mem.h>
21#include <dm/device_compat.h>
22#include <linux/io.h>
23
24#define SPI_IDLE_VAL 0xff
25
26#define MAX_CTRL_CS 7
27
28/* SPI_CONTROL */
29#define SPI_CTL_EN 0x00000001 /* Enable */
30#define SPI_CTL_MSTR 0x00000002 /* Master/Slave */
31#define SPI_CTL_PSSE 0x00000004 /* controls modf error in master mode */
32#define SPI_CTL_ODM 0x00000008 /* Open Drain Mode */
33#define SPI_CTL_CPHA 0x00000010 /* Clock Phase */
34#define SPI_CTL_CPOL 0x00000020 /* Clock Polarity */
35#define SPI_CTL_ASSEL 0x00000040 /* Slave Select Pin Control */
36#define SPI_CTL_SELST 0x00000080 /* Slave Select Polarity in transfers */
37#define SPI_CTL_EMISO 0x00000100 /*Enable MISO */
38#define SPI_CTL_SIZE 0x00000600 /*Word Transfer Size */
39#define SPI_CTL_SIZE08 0x00000000 /*SIZE: 8 bits */
40#define SPI_CTL_SIZE16 0x00000200 /*SIZE: 16 bits */
41#define SPI_CTL_SIZE32 0x00000400 /*SIZE: 32 bits */
42#define SPI_CTL_LSBF 0x00001000 /*LSB First */
43#define SPI_CTL_FCEN 0x00002000 /*Flow-Control Enable */
44#define SPI_CTL_FCCH 0x00004000 /*Flow-Control Channel Selection */
45#define SPI_CTL_FCPL 0x00008000 /*Flow-Control Polarity */
46#define SPI_CTL_FCWM 0x00030000 /*Flow-Control Water-Mark */
47#define SPI_CTL_FIFO0 0x00000000 /*FCWM: Tx empty or Rx Full */
48#define SPI_CTL_FIFO1 0x00010000 /*FCWM: Tx empty or Rx full (>=75%) */
49#define SPI_CTL_FIFO2 0x00020000 /*FCWM: Tx empty or Rx full (>=50%) */
50#define SPI_CTL_FMODE 0x00040000 /*Fast-mode Enable */
51#define SPI_CTL_MIOM 0x00300000 /*Multiple I/O Mode */
52#define SPI_CTL_MIO_DIS 0x00000000 /*MIOM: Disable */
53#define SPI_CTL_MIO_DUAL 0x00100000 /*MIOM: Enable DIOM (Dual I/O Mode) */
54#define SPI_CTL_MIO_QUAD 0x00200000 /*MIOM: Enable QUAD (Quad SPI Mode) */
55#define SPI_CTL_SOSI 0x00400000 /*Start on MOSI */
56#define SPI_CTL_MMWEM 0x40000000 /*Start on MMWEM */
57#define SPI_CTL_MMSE 0x80000000 /*Start on MMSE */
58/* SPI_RX_CONTROL */
59#define SPI_RXCTL_REN 0x00000001 /*Receive Channel Enable */
60#define SPI_RXCTL_RTI 0x00000004 /*Receive Transfer Initiate */
61#define SPI_RXCTL_RWCEN 0x00000008 /*Receive Word Counter Enable */
62#define SPI_RXCTL_RDR 0x00000070 /*Receive Data Request */
63#define SPI_RXCTL_RDR_DIS 0x00000000 /*RDR: Disabled */
64#define SPI_RXCTL_RDR_NE 0x00000010 /*RDR: RFIFO not empty */
65#define SPI_RXCTL_RDR_25 0x00000020 /*RDR: RFIFO 25% full */
66#define SPI_RXCTL_RDR_50 0x00000030 /*RDR: RFIFO 50% full */
67#define SPI_RXCTL_RDR_75 0x00000040 /*RDR: RFIFO 75% full */
68#define SPI_RXCTL_RDR_FULL 0x00000050 /*RDR: RFIFO full */
69#define SPI_RXCTL_RDO 0x00000100 /*Receive Data Over-Run */
70#define SPI_RXCTL_RRWM 0x00003000 /*FIFO Regular Water-Mark */
71#define SPI_RXCTL_RWM_0 0x00000000 /*RRWM: RFIFO Empty */
72#define SPI_RXCTL_RWM_25 0x00001000 /*RRWM: RFIFO 25% full */
73#define SPI_RXCTL_RWM_50 0x00002000 /*RRWM: RFIFO 50% full */
74#define SPI_RXCTL_RWM_75 0x00003000 /*RRWM: RFIFO 75% full */
75#define SPI_RXCTL_RUWM 0x00070000 /*FIFO Urgent Water-Mark */
76#define SPI_RXCTL_UWM_DIS 0x00000000 /*RUWM: Disabled */
77#define SPI_RXCTL_UWM_25 0x00010000 /*RUWM: RFIFO 25% full */
78#define SPI_RXCTL_UWM_50 0x00020000 /*RUWM: RFIFO 50% full */
79#define SPI_RXCTL_UWM_75 0x00030000 /*RUWM: RFIFO 75% full */
80#define SPI_RXCTL_UWM_FULL 0x00040000 /*RUWM: RFIFO full */
81/* SPI_TX_CONTROL */
82#define SPI_TXCTL_TEN 0x00000001 /*Transmit Channel Enable */
83#define SPI_TXCTL_TTI 0x00000004 /*Transmit Transfer Initiate */
84#define SPI_TXCTL_TWCEN 0x00000008 /*Transmit Word Counter Enable */
85#define SPI_TXCTL_TDR 0x00000070 /*Transmit Data Request */
86#define SPI_TXCTL_TDR_DIS 0x00000000 /*TDR: Disabled */
87#define SPI_TXCTL_TDR_NF 0x00000010 /*TDR: TFIFO not full */
88#define SPI_TXCTL_TDR_25 0x00000020 /*TDR: TFIFO 25% empty */
89#define SPI_TXCTL_TDR_50 0x00000030 /*TDR: TFIFO 50% empty */
90#define SPI_TXCTL_TDR_75 0x00000040 /*TDR: TFIFO 75% empty */
91#define SPI_TXCTL_TDR_EMPTY 0x00000050 /*TDR: TFIFO empty */
92#define SPI_TXCTL_TDU 0x00000100 /*Transmit Data Under-Run */
93#define SPI_TXCTL_TRWM 0x00003000 /*FIFO Regular Water-Mark */
94#define SPI_TXCTL_RWM_FULL 0x00000000 /*TRWM: TFIFO full */
95#define SPI_TXCTL_RWM_25 0x00001000 /*TRWM: TFIFO 25% empty */
96#define SPI_TXCTL_RWM_50 0x00002000 /*TRWM: TFIFO 50% empty */
97#define SPI_TXCTL_RWM_75 0x00003000 /*TRWM: TFIFO 75% empty */
98#define SPI_TXCTL_TUWM 0x00070000 /*FIFO Urgent Water-Mark */
99#define SPI_TXCTL_UWM_DIS 0x00000000 /*TUWM: Disabled */
100#define SPI_TXCTL_UWM_25 0x00010000 /*TUWM: TFIFO 25% empty */
101#define SPI_TXCTL_UWM_50 0x00020000 /*TUWM: TFIFO 50% empty */
102#define SPI_TXCTL_UWM_75 0x00030000 /*TUWM: TFIFO 75% empty */
103#define SPI_TXCTL_UWM_EMPTY 0x00040000 /*TUWM: TFIFO empty */
104/* SPI_CLOCK */
105#define SPI_CLK_BAUD 0x0000FFFF /*Baud Rate */
106/* SPI_DELAY */
107#define SPI_DLY_STOP 0x000000FF /*Transfer delay time */
108#define SPI_DLY_LEADX 0x00000100 /*Extended (1 SCK) LEAD Control */
109#define SPI_DLY_LAGX 0x00000200 /*Extended (1 SCK) LAG control */
110/* SPI_SSEL */
111#define SPI_SLVSEL_SSE1 0x00000002 /*SPISSEL1 Enable */
112#define SPI_SLVSEL_SSE2 0x00000004 /*SPISSEL2 Enable */
113#define SPI_SLVSEL_SSE3 0x00000008 /*SPISSEL3 Enable */
114#define SPI_SLVSEL_SSE4 0x00000010 /*SPISSEL4 Enable */
115#define SPI_SLVSEL_SSE5 0x00000020 /*SPISSEL5 Enable */
116#define SPI_SLVSEL_SSE6 0x00000040 /*SPISSEL6 Enable */
117#define SPI_SLVSEL_SSE7 0x00000080 /*SPISSEL7 Enable */
118#define SPI_SLVSEL_SSEL1 0x00000200 /*SPISSEL1 Value */
119#define SPI_SLVSEL_SSEL2 0x00000400 /*SPISSEL2 Value */
120#define SPI_SLVSEL_SSEL3 0x00000800 /*SPISSEL3 Value */
121#define SPI_SLVSEL_SSEL4 0x00001000 /*SPISSEL4 Value */
122#define SPI_SLVSEL_SSEL5 0x00002000 /*SPISSEL5 Value */
123#define SPI_SLVSEL_SSEL6 0x00004000 /*SPISSEL6 Value */
124#define SPI_SLVSEL_SSEL7 0x00008000 /*SPISSEL7 Value */
125/* SPI_RWC */
126#define SPI_RWC_VALUE 0x0000FFFF /*Received Word-Count */
127/* SPI_RWCR */
128#define SPI_RWCR_VALUE 0x0000FFFF /*Received Word-Count Reload */
129/* SPI_TWC */
130#define SPI_TWC_VALUE 0x0000FFFF /*Transmitted Word-Count */
131/* SPI_TWCR */
132#define SPI_TWCR_VALUE 0x0000FFFF /*Transmitted Word-Count Reload */
133/* SPI_IMASK */
134#define SPI_IMSK_RUWM 0x00000002 /*Receive Water-Mark Interrupt Mask */
135#define SPI_IMSK_TUWM 0x00000004 /*Transmit Water-Mark Interrupt Mask */
136#define SPI_IMSK_ROM 0x00000010 /*Receive Over-Run Interrupt Mask */
137#define SPI_IMSK_TUM 0x00000020 /*Transmit Under-Run Interrupt Mask */
138#define SPI_IMSK_TCM 0x00000040 /*Transmit Collision Interrupt Mask */
139#define SPI_IMSK_MFM 0x00000080 /*Mode Fault Interrupt Mask */
140#define SPI_IMSK_RSM 0x00000100 /*Receive Start Interrupt Mask */
141#define SPI_IMSK_TSM 0x00000200 /*Transmit Start Interrupt Mask */
142#define SPI_IMSK_RFM 0x00000400 /*Receive Finish Interrupt Mask */
143#define SPI_IMSK_TFM 0x00000800 /*Transmit Finish Interrupt Mask */
144/* SPI_IMASKCL */
145#define SPI_IMSK_CLR_RUW 0x00000002 /*Receive Water-Mark Interrupt Mask */
146#define SPI_IMSK_CLR_TUWM 0x00000004 /*Transmit Water-Mark Interrupt Mask */
147#define SPI_IMSK_CLR_ROM 0x00000010 /*Receive Over-Run Interrupt Mask */
148#define SPI_IMSK_CLR_TUM 0x00000020 /*Transmit Under-Run Interrupt Mask */
149#define SPI_IMSK_CLR_TCM 0x00000040 /*Transmit Collision Interrupt Mask */
150#define SPI_IMSK_CLR_MFM 0x00000080 /*Mode Fault Interrupt Mask */
151#define SPI_IMSK_CLR_RSM 0x00000100 /*Receive Start Interrupt Mask */
152#define SPI_IMSK_CLR_TSM 0x00000200 /*Transmit Start Interrupt Mask */
153#define SPI_IMSK_CLR_RFM 0x00000400 /*Receive Finish Interrupt Mask */
154#define SPI_IMSK_CLR_TFM 0x00000800 /*Transmit Finish Interrupt Mask */
155/* SPI_IMASKST */
156#define SPI_IMSK_SET_RUWM 0x00000002 /*Receive Water-Mark Interrupt Mask */
157#define SPI_IMSK_SET_TUWM 0x00000004 /*Transmit Water-Mark Interrupt Mask */
158#define SPI_IMSK_SET_ROM 0x00000010 /*Receive Over-Run Interrupt Mask */
159#define SPI_IMSK_SET_TUM 0x00000020 /*Transmit Under-Run Interrupt Mask */
160#define SPI_IMSK_SET_TCM 0x00000040 /*Transmit Collision Interrupt Mask */
161#define SPI_IMSK_SET_MFM 0x00000080 /*Mode Fault Interrupt Mask */
162#define SPI_IMSK_SET_RSM 0x00000100 /*Receive Start Interrupt Mask */
163#define SPI_IMSK_SET_TSM 0x00000200 /*Transmit Start Interrupt Mask */
164#define SPI_IMSK_SET_RFM 0x00000400 /*Receive Finish Interrupt Mask */
165#define SPI_IMSK_SET_TFM 0x00000800 /*Transmit Finish Interrupt Mask */
166/* SPI_STATUS */
167#define SPI_STAT_SPIF 0x00000001 /*SPI Finished */
168#define SPI_STAT_RUWM 0x00000002 /*Receive Water-Mark Breached */
169#define SPI_STAT_TUWM 0x00000004 /*Transmit Water-Mark Breached */
170#define SPI_STAT_ROE 0x00000010 /*Receive Over-Run Indication */
171#define SPI_STAT_TUE 0x00000020 /*Transmit Under-Run Indication */
172#define SPI_STAT_TCE 0x00000040 /*Transmit Collision Indication */
173#define SPI_STAT_MODF 0x00000080 /*Mode Fault Indication */
174#define SPI_STAT_RS 0x00000100 /*Receive Start Indication */
175#define SPI_STAT_TS 0x00000200 /*Transmit Start Indication */
176#define SPI_STAT_RF 0x00000400 /*Receive Finish Indication */
177#define SPI_STAT_TF 0x00000800 /*Transmit Finish Indication */
178#define SPI_STAT_RFS 0x00007000 /*SPI_RFIFO status */
179#define SPI_STAT_RFIFO_EMPTY 0x00000000 /*RFS: RFIFO Empty */
180#define SPI_STAT_RFIFO_25 0x00001000 /*RFS: RFIFO 25% Full */
181#define SPI_STAT_RFIFO_50 0x00002000 /*RFS: RFIFO 50% Full */
182#define SPI_STAT_RFIFO_75 0x00003000 /*RFS: RFIFO 75% Full */
183#define SPI_STAT_RFIFO_FULL 0x00004000 /*RFS: RFIFO Full */
184#define SPI_STAT_TFS 0x00070000 /*SPI_TFIFO status */
185#define SPI_STAT_TFIFO_FULL 0x00000000 /*TFS: TFIFO full */
186#define SPI_STAT_TFIFO_25 0x00010000 /*TFS: TFIFO 25% empty */
187#define SPI_STAT_TFIFO_50 0x00020000 /*TFS: TFIFO 50% empty */
188#define SPI_STAT_TFIFO_75 0x00030000 /*TFS: TFIFO 75% empty */
189#define SPI_STAT_TFIFO_EMPTY 0x00040000 /*TFS: TFIFO empty */
190#define SPI_STAT_FCS 0x00100000 /*Flow-Control Stall Indication */
191#define SPI_STAT_RFE 0x00400000 /*SPI_RFIFO Empty */
192#define SPI_STAT_TFF 0x00800000 /*SPI_TFIFO Full */
193/* SPI_ILAT */
194#define SPI_ILAT_RUWMI 0x00000002 /*Receive Water Mark Interrupt */
195#define SPI_ILAT_TUWMI 0x00000004 /*Transmit Water Mark Interrupt */
196#define SPI_ILAT_ROI 0x00000010 /*Receive Over-Run Indication */
197#define SPI_ILAT_TUI 0x00000020 /*Transmit Under-Run Indication */
198#define SPI_ILAT_TCI 0x00000040 /*Transmit Collision Indication */
199#define SPI_ILAT_MFI 0x00000080 /*Mode Fault Indication */
200#define SPI_ILAT_RSI 0x00000100 /*Receive Start Indication */
201#define SPI_ILAT_TSI 0x00000200 /*Transmit Start Indication */
202#define SPI_ILAT_RFI 0x00000400 /*Receive Finish Indication */
203#define SPI_ILAT_TFI 0x00000800 /*Transmit Finish Indication */
204/* SPI_ILATCL */
205#define SPI_ILAT_CLR_RUWMI 0x00000002 /*Receive Water Mark Interrupt */
206#define SPI_ILAT_CLR_TUWMI 0x00000004 /*Transmit Water Mark Interrupt */
207#define SPI_ILAT_CLR_ROI 0x00000010 /*Receive Over-Run Indication */
208#define SPI_ILAT_CLR_TUI 0x00000020 /*Transmit Under-Run Indication */
209#define SPI_ILAT_CLR_TCI 0x00000040 /*Transmit Collision Indication */
210#define SPI_ILAT_CLR_MFI 0x00000080 /*Mode Fault Indication */
211#define SPI_ILAT_CLR_RSI 0x00000100 /*Receive Start Indication */
212#define SPI_ILAT_CLR_TSI 0x00000200 /*Transmit Start Indication */
213#define SPI_ILAT_CLR_RFI 0x00000400 /*Receive Finish Indication */
214#define SPI_ILAT_CLR_TFI 0x00000800 /*Transmit Finish Indication */
215/* SPI_MMRDH */
216#define SPI_MMRDH_MERGE 0x04000000 /*Merge Enable */
217#define SPI_MMRDH_DMY_SZ 0x00007000 /*Bytes of Dummy */
218#define SPI_MMRDH_ADDR_PINS 0x00000800 /*Pins used for Address */
219#define SPI_MMRDH_ADDR_SZ 0x00000700 /*Bytes of Read Address */
220#define SPI_MMRDH_OPCODE 0x000000FF /*Read Opcode */
221
222#define SPI_MMRDH_TRIDMY_OFF 24 /*Bytes of Dummy offset */
223#define SPI_MMRDH_DMY_SZ_OFF 12 /*Bytes of Dummy offset */
224#define SPI_MMRDH_ADDR_SZ_OFF 8 /*Bytes of Read Address offset */
225
226#define BIT_SSEL_VAL(x) ((1 << 8) << (x)) /* Slave Select input value bit */
227#define BIT_SSEL_EN(x) (1 << (x)) /* Slave Select enable bit*/
228
229struct adi_spi_regs {
230 u32 revid;
231 u32 control;
232 u32 rx_control;
233 u32 tx_control;
234 u32 clock;
235 u32 delay;
236 u32 ssel;
237 u32 rwc;
238 u32 rwcr;
239 u32 twc;
240 u32 twcr;
241 u32 reserved0;
242 u32 emask;
243 u32 emaskcl;
244 u32 emaskst;
245 u32 reserved1;
246 u32 status;
247 u32 elat;
248 u32 elatcl;
249 u32 reserved2;
250 u32 rfifo;
251 u32 reserved3;
252 u32 tfifo;
253 u32 reserved4;
254 u32 mmrdh;
255 u32 mmtop;
256};
257
258struct adi_spi_platdata {
259 u32 max_hz;
260 u32 bus_num;
261 struct adi_spi_regs __iomem *regs;
262};
263
264struct adi_spi_priv {
265 u32 control;
266 u32 clock;
267 u32 bus_num;
268 u32 max_cs;
269 struct adi_spi_regs __iomem *regs;
270};
271
272/**
273 * By convention, this driver uses the same CS numbering that is used with the SSEL bit
274 * definitions (both here and in the TRM on which this is based), which are 1-indexed not
275 * 0-indexed. The valid CS range is therefore [1,max_cs], in contrast with other drivers
276 * where it is [0,max_cs-1].
277 */
278static int adi_spi_cs_info(struct udevice *bus, uint cs,
279 struct spi_cs_info *info)
280{
281 struct adi_spi_priv *priv = dev_get_priv(bus);
282
283 if (cs == 0 || cs > priv->max_cs) {
284 dev_err(bus, "invalid chipselect %u\n", cs);
285 return -EINVAL;
286 }
287
288 return 0;
289}
290
291static int adi_spi_of_to_plat(struct udevice *bus)
292{
293 struct adi_spi_platdata *plat = dev_get_plat(bus);
294 fdt_addr_t addr;
295
296 plat->max_hz = dev_read_u32_default(bus, "spi-max-frequency", 500000);
297 plat->bus_num = dev_read_u32_default(bus, "bus-num", 0);
298 addr = dev_read_addr(bus);
299
300 if (addr == FDT_ADDR_T_NONE)
301 return -EINVAL;
302
303 plat->regs = map_sysmem(addr, sizeof(*plat->regs));
304
305 return 0;
306}
307
308static int adi_spi_probe(struct udevice *bus)
309{
310 struct adi_spi_platdata *plat = dev_get_plat(bus);
311 struct adi_spi_priv *priv = dev_get_priv(bus);
312
313 priv->bus_num = plat->bus_num;
314 priv->regs = plat->regs;
315 priv->max_cs = dev_read_u32_default(bus, "num-cs", MAX_CTRL_CS);
316
317 iowrite32(0x0, &plat->regs->control);
318 iowrite32(0x0, &plat->regs->rx_control);
319 iowrite32(0x0, &plat->regs->tx_control);
320
321 return 0;
322}
323
324static int adi_spi_remove(struct udevice *dev)
325{
326 return -ENODEV;
327}
328
329static int adi_spi_claim_bus(struct udevice *dev)
330{
331 struct adi_spi_priv *priv;
332 struct udevice *bus = dev->parent;
333
334 priv = dev_get_priv(bus);
335
336 debug("%s: control:%i clock:%i\n",
337 __func__, priv->control, priv->clock);
338
339 iowrite32(priv->control, &priv->regs->control);
340 iowrite32(priv->clock, &priv->regs->clock);
341 iowrite32(0x0, &priv->regs->delay);
342
343 return 0;
344}
345
346static int adi_spi_release_bus(struct udevice *dev)
347{
348 struct adi_spi_priv *priv;
349 struct udevice *bus = dev->parent;
350
351 priv = dev_get_priv(bus);
352
353 debug("%s: control:%i clock:%i\n",
354 __func__, priv->control, priv->clock);
355
356 iowrite32(0x0, &priv->regs->rx_control);
357 iowrite32(0x0, &priv->regs->tx_control);
358 iowrite32(0x0, &priv->regs->control);
359
360 return 0;
361}
362
363void adi_spi_enable_ssel(struct adi_spi_priv *priv, int cs)
364{
365 setbits_32(&priv->regs->ssel, BIT_SSEL_EN(cs));
366}
367
368void adi_spi_set_ssel(struct adi_spi_priv *priv, int cs, int high)
369{
370 if (high)
371 setbits_32(&priv->regs->ssel, BIT_SSEL_VAL(cs));
372 else
373 clrbits_32(&priv->regs->ssel, BIT_SSEL_VAL(cs));
374}
375
376void adi_spi_cs_activate(struct adi_spi_priv *priv, struct dm_spi_slave_plat *slave_plat)
377{
378 bool high = slave_plat->mode & SPI_CS_HIGH;
379
380 adi_spi_set_ssel(priv, slave_plat->cs[0], high);
381 adi_spi_enable_ssel(priv, slave_plat->cs[0]);
382}
383
384void adi_spi_cs_deactivate(struct adi_spi_priv *priv, struct dm_spi_slave_plat *slave_plat)
385{
386 bool high = slave_plat->mode & SPI_CS_HIGH;
387
388 /* invert CS for matching SSEL to deactivate */
389 adi_spi_set_ssel(priv, slave_plat->cs[0], !high);
390}
391
392static void discard_rx_fifo_contents(struct adi_spi_regs *regs)
393{
394 while (!(ioread32(&regs->status) & SPI_STAT_RFE))
395 ioread32(&regs->rfifo);
396}
397
398static int adi_spi_fifo_mio_xfer(struct adi_spi_priv *priv, const u8 *tx, u8 *rx,
399 uint bytes, uint32_t mio_mode)
400{
401 u8 value;
402
403 /* switch current SPI transfer to mio SPI mode */
404 clrsetbits_32(&priv->regs->control, SPI_CTL_SOSI, mio_mode);
405 /*
406 * Data can only be transferred in one direction in multi-io SPI
407 * modes, trigger the transfer in respective direction.
408 */
409 if (rx) {
410 iowrite32(0x0, &priv->regs->tx_control);
411 iowrite32(SPI_RXCTL_REN | SPI_RXCTL_RTI, &priv->regs->rx_control);
412
413 while (bytes--) {
414 while (ioread32(&priv->regs->status) &
415 SPI_STAT_RFE)
416 if (ctrlc())
417 return -1;
418 value = ioread32(&priv->regs->rfifo);
419 *rx++ = value;
420 }
421 } else if (tx) {
422 iowrite32(0x0, &priv->regs->rx_control);
423 iowrite32(SPI_TXCTL_TEN | SPI_TXCTL_TTI, &priv->regs->tx_control);
424
425 while (bytes--) {
426 value = *tx++;
427 iowrite32(value, &priv->regs->tfifo);
428 while (ioread32(&priv->regs->status) &
429 SPI_STAT_TFF)
430 if (ctrlc())
431 return -1;
432 }
433
434 /* Wait till the tfifo is empty */
435 while ((ioread32(&priv->regs->status) & SPI_STAT_TFS) != SPI_STAT_TFIFO_EMPTY)
436 if (ctrlc())
437 return -1;
438 } else {
439 return -1;
440 }
441 return 0;
442}
443
444static int adi_spi_fifo_1x_xfer(struct adi_spi_priv *priv, const u8 *tx, u8 *rx,
445 uint bytes)
446{
447 u8 value;
448
449 /*
450 * Set current SPI transfer in normal mode and trigger
451 * the bi-direction transfer by tx write operation.
452 */
453 iowrite32(priv->control, &priv->regs->control);
454 iowrite32(SPI_RXCTL_REN, &priv->regs->rx_control);
455 iowrite32(SPI_TXCTL_TEN | SPI_TXCTL_TTI, &priv->regs->tx_control);
456
457 while (bytes--) {
458 value = (tx ? *tx++ : SPI_IDLE_VAL);
459 debug("%s: tx:%x ", __func__, value);
460 iowrite32(value, &priv->regs->tfifo);
461 while (ioread32(&priv->regs->status) & SPI_STAT_RFE)
462 if (ctrlc())
463 return -1;
464 value = ioread32(&priv->regs->rfifo);
465 if (rx)
466 *rx++ = value;
467 debug("rx:%x\n", value);
468 }
469 return 0;
470}
471
472static int adi_spi_fifo_xfer(struct adi_spi_priv *priv, int buswidth,
473 const u8 *tx, u8 *rx, uint bytes)
474{
475 switch (buswidth) {
476 case 1:
477 return adi_spi_fifo_1x_xfer(priv, tx, rx, bytes);
478 case 2:
479 return adi_spi_fifo_mio_xfer(priv, tx, rx, bytes, SPI_CTL_MIO_DUAL);
480 case 4:
481 return adi_spi_fifo_mio_xfer(priv, tx, rx, bytes, SPI_CTL_MIO_QUAD);
482 default:
483 return -ENOTSUPP;
484 }
485}
486
487static int adi_spi_xfer(struct udevice *dev, unsigned int bitlen,
488 const void *dout, void *din, unsigned long flags)
489{
490 struct udevice *bus = dev->parent;
491 struct adi_spi_priv *priv = dev_get_priv(bus);
492 struct dm_spi_slave_plat *slave_plat = dev_get_parent_plat(dev);
493
494 const u8 *tx = dout;
495 u8 *rx = din;
496 uint bytes = bitlen / 8;
497 int ret = 0;
498
499 debug("%s: bus_num:%i cs:%i bitlen:%i bytes:%i flags:%lx\n", __func__,
500 priv->bus_num, slave_plat->cs[0], bitlen, bytes, flags);
501
502 if (flags & SPI_XFER_BEGIN)
503 adi_spi_cs_activate(priv, slave_plat);
504
505 if (bitlen == 0)
506 goto done;
507
508 /* we can only do 8 bit transfers */
509 if (bitlen % 8) {
510 flags |= SPI_XFER_END;
511 goto done;
512 }
513
514 /* Discard invalid rx data and empty rfifo */
515 discard_rx_fifo_contents(priv->regs);
516
517 ret = adi_spi_fifo_1x_xfer(priv, tx, rx, bytes);
518
519 done:
520 if (flags & SPI_XFER_END)
521 adi_spi_cs_deactivate(priv, slave_plat);
522
523 return ret;
524}
525
526static int adi_spi_set_speed(struct udevice *bus, uint speed)
527{
528 struct adi_spi_platdata *plat = dev_get_plat(bus);
529 struct adi_spi_priv *priv = dev_get_priv(bus);
530 int ret;
531 u32 clock, spi_base_clk;
532 struct clk spi_clk;
533
534 ret = clk_get_by_name(bus, "spi", &spi_clk);
535 if (ret < 0) {
536 dev_err(bus, "Can't get SPI clk: %d\n", ret);
537 return ret;
538 }
539 spi_base_clk = clk_get_rate(&spi_clk);
540
541 if (speed > plat->max_hz)
542 speed = plat->max_hz;
543
544 if (speed > spi_base_clk)
545 return -ENODEV;
546
547 clock = spi_base_clk / speed;
548 if (clock)
549 clock--;
550
551 priv->clock = clock;
552
553 debug("%s: priv->clock: %x, speed: %x, get_spi_clk(): %x\n",
554 __func__, clock, speed, spi_base_clk);
555
556 return 0;
557}
558
559static int adi_spi_set_mode(struct udevice *bus, uint mode)
560{
561 struct adi_spi_priv *priv = dev_get_priv(bus);
562 u32 reg;
563
564 reg = SPI_CTL_EN | SPI_CTL_MSTR;
565 if (mode & SPI_CPHA)
566 reg |= SPI_CTL_CPHA;
567 if (mode & SPI_CPOL)
568 reg |= SPI_CTL_CPOL;
569 if (mode & SPI_LSB_FIRST)
570 reg |= SPI_CTL_LSBF;
571 reg &= ~SPI_CTL_ASSEL;
572
573 priv->control = reg;
574
575 debug("%s: control=%d, cs_pol=%d\n", __func__, reg, mode & SPI_CS_HIGH ? 1 : 0);
576
577 return 0;
578}
579
580/**
581 * U-boot's version of spi-mem does not support mixed bus-width
582 * commands nor anything more than 1x mode.
583 * Using a custom exec_op implementation, we can support it.
584 */
585static int adi_spi_mem_exec_op(struct spi_slave *slave,
586 const struct spi_mem_op *op)
587{
588 int rv = 0;
589 struct udevice *bus = slave->dev->parent;
590 struct adi_spi_priv *priv = dev_get_priv(bus);
591 struct dm_spi_slave_plat *slave_plat = dev_get_parent_plat(slave->dev);
592 u8 tmpbuf[64];
593 int i;
594
595 if ((op->cmd.nbytes + op->addr.nbytes + op->dummy.nbytes) >
596 sizeof(tmpbuf))
597 return -ENOMEM;
598
599 for (i = 0; i < op->cmd.nbytes; i++)
600 tmpbuf[i] = op->cmd.opcode >>
601 (8 * (op->cmd.nbytes - i - 1));
602 for (i = 0; i < op->addr.nbytes; i++)
603 tmpbuf[i + op->cmd.nbytes] = op->addr.val >>
604 (8 * (op->addr.nbytes - i - 1));
605 memset(tmpbuf + op->addr.nbytes + op->cmd.nbytes, 0xff,
606 op->dummy.nbytes);
607
608 adi_spi_cs_activate(priv, slave_plat);
609 discard_rx_fifo_contents(priv->regs);
610
611 if (op->cmd.nbytes) {
612 rv = adi_spi_fifo_xfer(priv, op->cmd.buswidth,
613 tmpbuf, NULL, op->cmd.nbytes);
614 if (rv != 0)
615 goto cleanup;
616 }
617
618 if (op->addr.nbytes) {
619 rv = adi_spi_fifo_xfer(priv, op->addr.buswidth,
620 tmpbuf + op->cmd.nbytes, NULL,
621 op->addr.nbytes);
622 if (rv != 0)
623 goto cleanup;
624 }
625
626 if (op->dummy.nbytes) {
627 rv = adi_spi_fifo_xfer(priv, op->dummy.buswidth,
628 tmpbuf + op->cmd.nbytes +
629 op->addr.nbytes,
630 NULL, op->dummy.nbytes);
631 if (rv != 0)
632 goto cleanup;
633 }
634
635 if (op->data.dir == SPI_MEM_DATA_IN)
636 rv = adi_spi_fifo_xfer(priv, op->data.buswidth,
637 NULL, op->data.buf.in,
638 op->data.nbytes);
639 else if (op->data.dir == SPI_MEM_DATA_OUT)
640 rv = adi_spi_fifo_xfer(priv, op->data.buswidth,
641 op->data.buf.out, NULL,
642 op->data.nbytes);
643
644cleanup:
645 adi_spi_cs_deactivate(priv, slave_plat);
646 return rv;
647}
648
649static const struct spi_controller_mem_ops adi_spi_mem_ops = {
650 .exec_op = adi_spi_mem_exec_op,
651};
652
653static const struct dm_spi_ops adi_spi_ops = {
654 .claim_bus = adi_spi_claim_bus,
655 .release_bus = adi_spi_release_bus,
656 .xfer = adi_spi_xfer,
657 .set_speed = adi_spi_set_speed,
658 .set_mode = adi_spi_set_mode,
659 .cs_info = adi_spi_cs_info,
660 .mem_ops = &adi_spi_mem_ops,
661};
662
663static const struct udevice_id adi_spi_ids[] = {
664 { .compatible = "adi,spi3" },
665 { }
666};
667
668U_BOOT_DRIVER(adi_spi3) = {
669 .name = "adi_spi3",
670 .id = UCLASS_SPI,
671 .of_match = adi_spi_ids,
672 .ops = &adi_spi_ops,
673 .of_to_plat = adi_spi_of_to_plat,
674 .probe = adi_spi_probe,
675 .remove = adi_spi_remove,
676 .plat_auto = sizeof(struct adi_spi_platdata),
677 .priv_auto = sizeof(struct adi_spi_priv),
678 .per_child_auto = sizeof(struct spi_slave),
679};