xref: /openbmc/qemu/hw/ssi/imx_spi.c (revision 28ae3179fc52d2e4d870b635c4a412aab99759e7)
1c906a3a0SJean-Christophe DUBOIS /*
2c906a3a0SJean-Christophe DUBOIS  * IMX SPI Controller
3c906a3a0SJean-Christophe DUBOIS  *
4c906a3a0SJean-Christophe DUBOIS  * Copyright (c) 2016 Jean-Christophe Dubois <jcd@tribudubois.net>
5c906a3a0SJean-Christophe DUBOIS  *
6c906a3a0SJean-Christophe DUBOIS  * This work is licensed under the terms of the GNU GPL, version 2 or later.
7c906a3a0SJean-Christophe DUBOIS  * See the COPYING file in the top-level directory.
8c906a3a0SJean-Christophe DUBOIS  *
9c906a3a0SJean-Christophe DUBOIS  */
10c906a3a0SJean-Christophe DUBOIS 
11c906a3a0SJean-Christophe DUBOIS #include "qemu/osdep.h"
1264552b6bSMarkus Armbruster #include "hw/irq.h"
13c906a3a0SJean-Christophe DUBOIS #include "hw/ssi/imx_spi.h"
14d6454270SMarkus Armbruster #include "migration/vmstate.h"
1503dd024fSPaolo Bonzini #include "qemu/log.h"
160b8fa32fSMarkus Armbruster #include "qemu/module.h"
17c906a3a0SJean-Christophe DUBOIS 
18c906a3a0SJean-Christophe DUBOIS #ifndef DEBUG_IMX_SPI
19c906a3a0SJean-Christophe DUBOIS #define DEBUG_IMX_SPI 0
20c906a3a0SJean-Christophe DUBOIS #endif
21c906a3a0SJean-Christophe DUBOIS 
22c906a3a0SJean-Christophe DUBOIS #define DPRINTF(fmt, args...) \
23c906a3a0SJean-Christophe DUBOIS     do { \
24c906a3a0SJean-Christophe DUBOIS         if (DEBUG_IMX_SPI) { \
25c906a3a0SJean-Christophe DUBOIS             fprintf(stderr, "[%s]%s: " fmt , TYPE_IMX_SPI, \
26c906a3a0SJean-Christophe DUBOIS                                              __func__, ##args); \
27c906a3a0SJean-Christophe DUBOIS         } \
28c906a3a0SJean-Christophe DUBOIS     } while (0)
29c906a3a0SJean-Christophe DUBOIS 
imx_spi_reg_name(uint32_t reg)30d675765aSPeter Maydell static const char *imx_spi_reg_name(uint32_t reg)
31c906a3a0SJean-Christophe DUBOIS {
32c906a3a0SJean-Christophe DUBOIS     static char unknown[20];
33c906a3a0SJean-Christophe DUBOIS 
34c906a3a0SJean-Christophe DUBOIS     switch (reg) {
35c906a3a0SJean-Christophe DUBOIS     case ECSPI_RXDATA:
36c906a3a0SJean-Christophe DUBOIS         return  "ECSPI_RXDATA";
37c906a3a0SJean-Christophe DUBOIS     case ECSPI_TXDATA:
38c906a3a0SJean-Christophe DUBOIS         return  "ECSPI_TXDATA";
39c906a3a0SJean-Christophe DUBOIS     case ECSPI_CONREG:
40c906a3a0SJean-Christophe DUBOIS         return  "ECSPI_CONREG";
41c906a3a0SJean-Christophe DUBOIS     case ECSPI_CONFIGREG:
42c906a3a0SJean-Christophe DUBOIS         return  "ECSPI_CONFIGREG";
43c906a3a0SJean-Christophe DUBOIS     case ECSPI_INTREG:
44c906a3a0SJean-Christophe DUBOIS         return  "ECSPI_INTREG";
45c906a3a0SJean-Christophe DUBOIS     case ECSPI_DMAREG:
46c906a3a0SJean-Christophe DUBOIS         return  "ECSPI_DMAREG";
47c906a3a0SJean-Christophe DUBOIS     case ECSPI_STATREG:
48c906a3a0SJean-Christophe DUBOIS         return  "ECSPI_STATREG";
49c906a3a0SJean-Christophe DUBOIS     case ECSPI_PERIODREG:
50c906a3a0SJean-Christophe DUBOIS         return  "ECSPI_PERIODREG";
51c906a3a0SJean-Christophe DUBOIS     case ECSPI_TESTREG:
52c906a3a0SJean-Christophe DUBOIS         return  "ECSPI_TESTREG";
53c906a3a0SJean-Christophe DUBOIS     case ECSPI_MSGDATA:
54c906a3a0SJean-Christophe DUBOIS         return  "ECSPI_MSGDATA";
55c906a3a0SJean-Christophe DUBOIS     default:
56ca4af17cSPhilippe Mathieu-Daudé         snprintf(unknown, sizeof(unknown), "%u ?", reg);
57c906a3a0SJean-Christophe DUBOIS         return unknown;
58c906a3a0SJean-Christophe DUBOIS     }
59c906a3a0SJean-Christophe DUBOIS }
60c906a3a0SJean-Christophe DUBOIS 
61c906a3a0SJean-Christophe DUBOIS static const VMStateDescription vmstate_imx_spi = {
62c906a3a0SJean-Christophe DUBOIS     .name = TYPE_IMX_SPI,
63c906a3a0SJean-Christophe DUBOIS     .version_id = 1,
64c906a3a0SJean-Christophe DUBOIS     .minimum_version_id = 1,
650aa6c7dfSRichard Henderson     .fields = (const VMStateField[]) {
66c906a3a0SJean-Christophe DUBOIS         VMSTATE_FIFO32(tx_fifo, IMXSPIState),
67c906a3a0SJean-Christophe DUBOIS         VMSTATE_FIFO32(rx_fifo, IMXSPIState),
68c906a3a0SJean-Christophe DUBOIS         VMSTATE_INT16(burst_length, IMXSPIState),
69c906a3a0SJean-Christophe DUBOIS         VMSTATE_UINT32_ARRAY(regs, IMXSPIState, ECSPI_MAX),
70c906a3a0SJean-Christophe DUBOIS         VMSTATE_END_OF_LIST()
71c906a3a0SJean-Christophe DUBOIS     },
72c906a3a0SJean-Christophe DUBOIS };
73c906a3a0SJean-Christophe DUBOIS 
imx_spi_txfifo_reset(IMXSPIState * s)74c906a3a0SJean-Christophe DUBOIS static void imx_spi_txfifo_reset(IMXSPIState *s)
75c906a3a0SJean-Christophe DUBOIS {
76c906a3a0SJean-Christophe DUBOIS     fifo32_reset(&s->tx_fifo);
77c906a3a0SJean-Christophe DUBOIS     s->regs[ECSPI_STATREG] |= ECSPI_STATREG_TE;
78c906a3a0SJean-Christophe DUBOIS     s->regs[ECSPI_STATREG] &= ~ECSPI_STATREG_TF;
79c906a3a0SJean-Christophe DUBOIS }
80c906a3a0SJean-Christophe DUBOIS 
imx_spi_rxfifo_reset(IMXSPIState * s)81c906a3a0SJean-Christophe DUBOIS static void imx_spi_rxfifo_reset(IMXSPIState *s)
82c906a3a0SJean-Christophe DUBOIS {
83c906a3a0SJean-Christophe DUBOIS     fifo32_reset(&s->rx_fifo);
84c906a3a0SJean-Christophe DUBOIS     s->regs[ECSPI_STATREG] &= ~ECSPI_STATREG_RR;
85c906a3a0SJean-Christophe DUBOIS     s->regs[ECSPI_STATREG] &= ~ECSPI_STATREG_RF;
86c906a3a0SJean-Christophe DUBOIS     s->regs[ECSPI_STATREG] &= ~ECSPI_STATREG_RO;
87c906a3a0SJean-Christophe DUBOIS }
88c906a3a0SJean-Christophe DUBOIS 
imx_spi_update_irq(IMXSPIState * s)89c906a3a0SJean-Christophe DUBOIS static void imx_spi_update_irq(IMXSPIState *s)
90c906a3a0SJean-Christophe DUBOIS {
91c906a3a0SJean-Christophe DUBOIS     int level;
92c906a3a0SJean-Christophe DUBOIS 
93c906a3a0SJean-Christophe DUBOIS     if (fifo32_is_empty(&s->rx_fifo)) {
94c906a3a0SJean-Christophe DUBOIS         s->regs[ECSPI_STATREG] &= ~ECSPI_STATREG_RR;
95c906a3a0SJean-Christophe DUBOIS     } else {
96c906a3a0SJean-Christophe DUBOIS         s->regs[ECSPI_STATREG] |= ECSPI_STATREG_RR;
97c906a3a0SJean-Christophe DUBOIS     }
98c906a3a0SJean-Christophe DUBOIS 
99c906a3a0SJean-Christophe DUBOIS     if (fifo32_is_full(&s->rx_fifo)) {
100c906a3a0SJean-Christophe DUBOIS         s->regs[ECSPI_STATREG] |= ECSPI_STATREG_RF;
101c906a3a0SJean-Christophe DUBOIS     } else {
102c906a3a0SJean-Christophe DUBOIS         s->regs[ECSPI_STATREG] &= ~ECSPI_STATREG_RF;
103c906a3a0SJean-Christophe DUBOIS     }
104c906a3a0SJean-Christophe DUBOIS 
105c906a3a0SJean-Christophe DUBOIS     if (fifo32_is_empty(&s->tx_fifo)) {
106c906a3a0SJean-Christophe DUBOIS         s->regs[ECSPI_STATREG] |= ECSPI_STATREG_TE;
107c906a3a0SJean-Christophe DUBOIS     } else {
108c906a3a0SJean-Christophe DUBOIS         s->regs[ECSPI_STATREG] &= ~ECSPI_STATREG_TE;
109c906a3a0SJean-Christophe DUBOIS     }
110c906a3a0SJean-Christophe DUBOIS 
111c906a3a0SJean-Christophe DUBOIS     if (fifo32_is_full(&s->tx_fifo)) {
112c906a3a0SJean-Christophe DUBOIS         s->regs[ECSPI_STATREG] |= ECSPI_STATREG_TF;
113c906a3a0SJean-Christophe DUBOIS     } else {
114c906a3a0SJean-Christophe DUBOIS         s->regs[ECSPI_STATREG] &= ~ECSPI_STATREG_TF;
115c906a3a0SJean-Christophe DUBOIS     }
116c906a3a0SJean-Christophe DUBOIS 
117c906a3a0SJean-Christophe DUBOIS     level = s->regs[ECSPI_STATREG] & s->regs[ECSPI_INTREG] ? 1 : 0;
118c906a3a0SJean-Christophe DUBOIS 
119c906a3a0SJean-Christophe DUBOIS     qemu_set_irq(s->irq, level);
120c906a3a0SJean-Christophe DUBOIS 
121c906a3a0SJean-Christophe DUBOIS     DPRINTF("IRQ level is %d\n", level);
122c906a3a0SJean-Christophe DUBOIS }
123c906a3a0SJean-Christophe DUBOIS 
imx_spi_selected_channel(IMXSPIState * s)124c906a3a0SJean-Christophe DUBOIS static uint8_t imx_spi_selected_channel(IMXSPIState *s)
125c906a3a0SJean-Christophe DUBOIS {
126c906a3a0SJean-Christophe DUBOIS     return EXTRACT(s->regs[ECSPI_CONREG], ECSPI_CONREG_CHANNEL_SELECT);
127c906a3a0SJean-Christophe DUBOIS }
128c906a3a0SJean-Christophe DUBOIS 
imx_spi_burst_length(IMXSPIState * s)129c906a3a0SJean-Christophe DUBOIS static uint32_t imx_spi_burst_length(IMXSPIState *s)
130c906a3a0SJean-Christophe DUBOIS {
13124bf8ef3SBin Meng     uint32_t burst;
13224bf8ef3SBin Meng 
13324bf8ef3SBin Meng     burst = EXTRACT(s->regs[ECSPI_CONREG], ECSPI_CONREG_BURST_LENGTH) + 1;
13424bf8ef3SBin Meng     if (burst % 8) {
13524bf8ef3SBin Meng         burst = ROUND_UP(burst, 8);
13624bf8ef3SBin Meng     }
13724bf8ef3SBin Meng 
13824bf8ef3SBin Meng     return burst;
139c906a3a0SJean-Christophe DUBOIS }
140c906a3a0SJean-Christophe DUBOIS 
imx_spi_is_enabled(IMXSPIState * s)141c906a3a0SJean-Christophe DUBOIS static bool imx_spi_is_enabled(IMXSPIState *s)
142c906a3a0SJean-Christophe DUBOIS {
143c906a3a0SJean-Christophe DUBOIS     return s->regs[ECSPI_CONREG] & ECSPI_CONREG_EN;
144c906a3a0SJean-Christophe DUBOIS }
145c906a3a0SJean-Christophe DUBOIS 
imx_spi_channel_is_master(IMXSPIState * s)146c906a3a0SJean-Christophe DUBOIS static bool imx_spi_channel_is_master(IMXSPIState *s)
147c906a3a0SJean-Christophe DUBOIS {
148c906a3a0SJean-Christophe DUBOIS     uint8_t mode = EXTRACT(s->regs[ECSPI_CONREG], ECSPI_CONREG_CHANNEL_MODE);
149c906a3a0SJean-Christophe DUBOIS 
150c906a3a0SJean-Christophe DUBOIS     return (mode & (1 << imx_spi_selected_channel(s))) ? true : false;
151c906a3a0SJean-Christophe DUBOIS }
152c906a3a0SJean-Christophe DUBOIS 
imx_spi_is_multiple_master_burst(IMXSPIState * s)153c906a3a0SJean-Christophe DUBOIS static bool imx_spi_is_multiple_master_burst(IMXSPIState *s)
154c906a3a0SJean-Christophe DUBOIS {
155c906a3a0SJean-Christophe DUBOIS     uint8_t wave = EXTRACT(s->regs[ECSPI_CONFIGREG], ECSPI_CONFIGREG_SS_CTL);
156c906a3a0SJean-Christophe DUBOIS 
157c906a3a0SJean-Christophe DUBOIS     return imx_spi_channel_is_master(s) &&
158c906a3a0SJean-Christophe DUBOIS            !(s->regs[ECSPI_CONREG] & ECSPI_CONREG_SMC) &&
159c906a3a0SJean-Christophe DUBOIS            ((wave & (1 << imx_spi_selected_channel(s))) ? true : false);
160c906a3a0SJean-Christophe DUBOIS }
161c906a3a0SJean-Christophe DUBOIS 
imx_spi_flush_txfifo(IMXSPIState * s)162c906a3a0SJean-Christophe DUBOIS static void imx_spi_flush_txfifo(IMXSPIState *s)
163c906a3a0SJean-Christophe DUBOIS {
164c906a3a0SJean-Christophe DUBOIS     uint32_t tx;
165c906a3a0SJean-Christophe DUBOIS     uint32_t rx;
166c906a3a0SJean-Christophe DUBOIS 
167c906a3a0SJean-Christophe DUBOIS     DPRINTF("Begin: TX Fifo Size = %d, RX Fifo Size = %d\n",
168c906a3a0SJean-Christophe DUBOIS             fifo32_num_used(&s->tx_fifo), fifo32_num_used(&s->rx_fifo));
169c906a3a0SJean-Christophe DUBOIS 
170c906a3a0SJean-Christophe DUBOIS     while (!fifo32_is_empty(&s->tx_fifo)) {
171c906a3a0SJean-Christophe DUBOIS         int tx_burst = 0;
172c906a3a0SJean-Christophe DUBOIS 
173c906a3a0SJean-Christophe DUBOIS         if (s->burst_length <= 0) {
174c906a3a0SJean-Christophe DUBOIS             s->burst_length = imx_spi_burst_length(s);
175c906a3a0SJean-Christophe DUBOIS 
176c906a3a0SJean-Christophe DUBOIS             DPRINTF("Burst length = %d\n", s->burst_length);
177c906a3a0SJean-Christophe DUBOIS 
178c906a3a0SJean-Christophe DUBOIS             if (imx_spi_is_multiple_master_burst(s)) {
179c906a3a0SJean-Christophe DUBOIS                 s->regs[ECSPI_CONREG] |= ECSPI_CONREG_XCH;
180c906a3a0SJean-Christophe DUBOIS             }
181c906a3a0SJean-Christophe DUBOIS         }
182c906a3a0SJean-Christophe DUBOIS 
183c906a3a0SJean-Christophe DUBOIS         tx = fifo32_pop(&s->tx_fifo);
184c906a3a0SJean-Christophe DUBOIS 
185c906a3a0SJean-Christophe DUBOIS         DPRINTF("data tx:0x%08x\n", tx);
186c906a3a0SJean-Christophe DUBOIS 
1876ed92482SBin Meng         tx_burst = (s->burst_length % 32) ? : 32;
188c906a3a0SJean-Christophe DUBOIS 
189c906a3a0SJean-Christophe DUBOIS         rx = 0;
190c906a3a0SJean-Christophe DUBOIS 
1919c49c83eSEden Mikitas         while (tx_burst > 0) {
1928c495d13SBin Meng             uint8_t byte = tx >> (tx_burst - 8);
193c906a3a0SJean-Christophe DUBOIS 
194c906a3a0SJean-Christophe DUBOIS             DPRINTF("writing 0x%02x\n", (uint32_t)byte);
195c906a3a0SJean-Christophe DUBOIS 
196c906a3a0SJean-Christophe DUBOIS             /* We need to write one byte at a time */
197c906a3a0SJean-Christophe DUBOIS             byte = ssi_transfer(s->bus, byte);
198c906a3a0SJean-Christophe DUBOIS 
199c906a3a0SJean-Christophe DUBOIS             DPRINTF("0x%02x read\n", (uint32_t)byte);
200c906a3a0SJean-Christophe DUBOIS 
2018c495d13SBin Meng             rx = (rx << 8) | byte;
202c906a3a0SJean-Christophe DUBOIS 
203c906a3a0SJean-Christophe DUBOIS             /* Remove 8 bits from the actual burst */
204c906a3a0SJean-Christophe DUBOIS             tx_burst -= 8;
205c906a3a0SJean-Christophe DUBOIS             s->burst_length -= 8;
206c906a3a0SJean-Christophe DUBOIS         }
207c906a3a0SJean-Christophe DUBOIS 
208c906a3a0SJean-Christophe DUBOIS         DPRINTF("data rx:0x%08x\n", rx);
209c906a3a0SJean-Christophe DUBOIS 
210c906a3a0SJean-Christophe DUBOIS         if (fifo32_is_full(&s->rx_fifo)) {
211c906a3a0SJean-Christophe DUBOIS             s->regs[ECSPI_STATREG] |= ECSPI_STATREG_RO;
212c906a3a0SJean-Christophe DUBOIS         } else {
2136d686145SEden Mikitas             fifo32_push(&s->rx_fifo, rx);
214c906a3a0SJean-Christophe DUBOIS         }
215c906a3a0SJean-Christophe DUBOIS 
216c906a3a0SJean-Christophe DUBOIS         if (s->burst_length <= 0) {
217c906a3a0SJean-Christophe DUBOIS             if (!imx_spi_is_multiple_master_burst(s)) {
218c906a3a0SJean-Christophe DUBOIS                 s->regs[ECSPI_STATREG] |= ECSPI_STATREG_TC;
219c906a3a0SJean-Christophe DUBOIS                 break;
220c906a3a0SJean-Christophe DUBOIS             }
221c906a3a0SJean-Christophe DUBOIS         }
222c906a3a0SJean-Christophe DUBOIS     }
223c906a3a0SJean-Christophe DUBOIS 
224c906a3a0SJean-Christophe DUBOIS     if (fifo32_is_empty(&s->tx_fifo)) {
225c906a3a0SJean-Christophe DUBOIS         s->regs[ECSPI_STATREG] |= ECSPI_STATREG_TC;
226016d4b01STrent Piepho         s->regs[ECSPI_CONREG] &= ~ECSPI_CONREG_XCH;
227c906a3a0SJean-Christophe DUBOIS     }
228c906a3a0SJean-Christophe DUBOIS 
229c906a3a0SJean-Christophe DUBOIS     /* TODO: We should also use TDR and RDR bits */
230c906a3a0SJean-Christophe DUBOIS 
231c906a3a0SJean-Christophe DUBOIS     DPRINTF("End: TX Fifo Size = %d, RX Fifo Size = %d\n",
232c906a3a0SJean-Christophe DUBOIS             fifo32_num_used(&s->tx_fifo), fifo32_num_used(&s->rx_fifo));
233c906a3a0SJean-Christophe DUBOIS }
234c906a3a0SJean-Christophe DUBOIS 
imx_spi_common_reset(IMXSPIState * s)23593722b6fSPhilippe Mathieu-Daudé static void imx_spi_common_reset(IMXSPIState *s)
236c906a3a0SJean-Christophe DUBOIS {
23793722b6fSPhilippe Mathieu-Daudé     int i;
238c906a3a0SJean-Christophe DUBOIS 
23993722b6fSPhilippe Mathieu-Daudé     for (i = 0; i < ARRAY_SIZE(s->regs); i++) {
24093722b6fSPhilippe Mathieu-Daudé         switch (i) {
24193722b6fSPhilippe Mathieu-Daudé         case ECSPI_CONREG:
24293722b6fSPhilippe Mathieu-Daudé             /* CONREG is not updated on soft reset */
24393722b6fSPhilippe Mathieu-Daudé             break;
24493722b6fSPhilippe Mathieu-Daudé         case ECSPI_STATREG:
24593722b6fSPhilippe Mathieu-Daudé             s->regs[i] = 0x00000003;
24693722b6fSPhilippe Mathieu-Daudé             break;
24793722b6fSPhilippe Mathieu-Daudé         default:
24893722b6fSPhilippe Mathieu-Daudé             s->regs[i] = 0;
24993722b6fSPhilippe Mathieu-Daudé             break;
25093722b6fSPhilippe Mathieu-Daudé         }
25193722b6fSPhilippe Mathieu-Daudé     }
252c906a3a0SJean-Christophe DUBOIS 
253c906a3a0SJean-Christophe DUBOIS     imx_spi_rxfifo_reset(s);
254c906a3a0SJean-Christophe DUBOIS     imx_spi_txfifo_reset(s);
255c906a3a0SJean-Christophe DUBOIS 
256c906a3a0SJean-Christophe DUBOIS     s->burst_length = 0;
257c906a3a0SJean-Christophe DUBOIS }
258c906a3a0SJean-Christophe DUBOIS 
imx_spi_soft_reset(IMXSPIState * s)2593c9829e5SBin Meng static void imx_spi_soft_reset(IMXSPIState *s)
2603c9829e5SBin Meng {
26150dc2593SXuzhou Cheng     int i;
26250dc2593SXuzhou Cheng 
26393722b6fSPhilippe Mathieu-Daudé     imx_spi_common_reset(s);
2643c9829e5SBin Meng 
2653c9829e5SBin Meng     imx_spi_update_irq(s);
26650dc2593SXuzhou Cheng 
26750dc2593SXuzhou Cheng     for (i = 0; i < ECSPI_NUM_CS; i++) {
26850dc2593SXuzhou Cheng         qemu_set_irq(s->cs_lines[i], 1);
26950dc2593SXuzhou Cheng     }
2703c9829e5SBin Meng }
2713c9829e5SBin Meng 
imx_spi_reset(DeviceState * dev)27293722b6fSPhilippe Mathieu-Daudé static void imx_spi_reset(DeviceState *dev)
27393722b6fSPhilippe Mathieu-Daudé {
27493722b6fSPhilippe Mathieu-Daudé     IMXSPIState *s = IMX_SPI(dev);
27593722b6fSPhilippe Mathieu-Daudé 
27693722b6fSPhilippe Mathieu-Daudé     imx_spi_common_reset(s);
27793722b6fSPhilippe Mathieu-Daudé     s->regs[ECSPI_CONREG] = 0;
27893722b6fSPhilippe Mathieu-Daudé }
27993722b6fSPhilippe Mathieu-Daudé 
imx_spi_read(void * opaque,hwaddr offset,unsigned size)280c906a3a0SJean-Christophe DUBOIS static uint64_t imx_spi_read(void *opaque, hwaddr offset, unsigned size)
281c906a3a0SJean-Christophe DUBOIS {
282c906a3a0SJean-Christophe DUBOIS     uint32_t value = 0;
283c906a3a0SJean-Christophe DUBOIS     IMXSPIState *s = opaque;
284c906a3a0SJean-Christophe DUBOIS     uint32_t index = offset >> 2;
285c906a3a0SJean-Christophe DUBOIS 
286c906a3a0SJean-Christophe DUBOIS     if (index >=  ECSPI_MAX) {
287c906a3a0SJean-Christophe DUBOIS         qemu_log_mask(LOG_GUEST_ERROR, "[%s]%s: Bad register at offset 0x%"
288c906a3a0SJean-Christophe DUBOIS                       HWADDR_PRIx "\n", TYPE_IMX_SPI, __func__, offset);
289c906a3a0SJean-Christophe DUBOIS         return 0;
290c906a3a0SJean-Christophe DUBOIS     }
291c906a3a0SJean-Christophe DUBOIS 
2927c87bb53SPhilippe Mathieu-Daudé     value = s->regs[index];
2937c87bb53SPhilippe Mathieu-Daudé 
2947c87bb53SPhilippe Mathieu-Daudé     if (imx_spi_is_enabled(s)) {
295c906a3a0SJean-Christophe DUBOIS         switch (index) {
296c906a3a0SJean-Christophe DUBOIS         case ECSPI_RXDATA:
2977c87bb53SPhilippe Mathieu-Daudé             if (fifo32_is_empty(&s->rx_fifo)) {
298c906a3a0SJean-Christophe DUBOIS                 /* value is undefined */
299c906a3a0SJean-Christophe DUBOIS                 value = 0xdeadbeef;
300c906a3a0SJean-Christophe DUBOIS             } else {
301c906a3a0SJean-Christophe DUBOIS                 /* read from the RX FIFO */
302c906a3a0SJean-Christophe DUBOIS                 value = fifo32_pop(&s->rx_fifo);
303c906a3a0SJean-Christophe DUBOIS             }
304c906a3a0SJean-Christophe DUBOIS             break;
305c906a3a0SJean-Christophe DUBOIS         case ECSPI_TXDATA:
3067c87bb53SPhilippe Mathieu-Daudé             qemu_log_mask(LOG_GUEST_ERROR,
3077c87bb53SPhilippe Mathieu-Daudé                           "[%s]%s: Trying to read from TX FIFO\n",
308c906a3a0SJean-Christophe DUBOIS                           TYPE_IMX_SPI, __func__);
309c906a3a0SJean-Christophe DUBOIS 
310c906a3a0SJean-Christophe DUBOIS             /* Reading from TXDATA gives 0 */
311c906a3a0SJean-Christophe DUBOIS             break;
312c906a3a0SJean-Christophe DUBOIS         case ECSPI_MSGDATA:
3137c87bb53SPhilippe Mathieu-Daudé             qemu_log_mask(LOG_GUEST_ERROR,
3147c87bb53SPhilippe Mathieu-Daudé                           "[%s]%s: Trying to read from MSG FIFO\n",
315c906a3a0SJean-Christophe DUBOIS                           TYPE_IMX_SPI, __func__);
316c906a3a0SJean-Christophe DUBOIS             /* Reading from MSGDATA gives 0 */
317c906a3a0SJean-Christophe DUBOIS             break;
318c906a3a0SJean-Christophe DUBOIS         default:
319c906a3a0SJean-Christophe DUBOIS             break;
320c906a3a0SJean-Christophe DUBOIS         }
321c906a3a0SJean-Christophe DUBOIS 
322c906a3a0SJean-Christophe DUBOIS         imx_spi_update_irq(s);
3237c87bb53SPhilippe Mathieu-Daudé     }
3247c87bb53SPhilippe Mathieu-Daudé     DPRINTF("reg[%s] => 0x%" PRIx32 "\n", imx_spi_reg_name(index), value);
325c906a3a0SJean-Christophe DUBOIS 
326c906a3a0SJean-Christophe DUBOIS     return (uint64_t)value;
327c906a3a0SJean-Christophe DUBOIS }
328c906a3a0SJean-Christophe DUBOIS 
imx_spi_write(void * opaque,hwaddr offset,uint64_t value,unsigned size)329c906a3a0SJean-Christophe DUBOIS static void imx_spi_write(void *opaque, hwaddr offset, uint64_t value,
330c906a3a0SJean-Christophe DUBOIS                            unsigned size)
331c906a3a0SJean-Christophe DUBOIS {
332c906a3a0SJean-Christophe DUBOIS     IMXSPIState *s = opaque;
333c906a3a0SJean-Christophe DUBOIS     uint32_t index = offset >> 2;
334c906a3a0SJean-Christophe DUBOIS     uint32_t change_mask;
33524bf8ef3SBin Meng     uint32_t burst;
336c906a3a0SJean-Christophe DUBOIS 
337c906a3a0SJean-Christophe DUBOIS     if (index >=  ECSPI_MAX) {
338c906a3a0SJean-Christophe DUBOIS         qemu_log_mask(LOG_GUEST_ERROR, "[%s]%s: Bad register at offset 0x%"
339c906a3a0SJean-Christophe DUBOIS                       HWADDR_PRIx "\n", TYPE_IMX_SPI, __func__, offset);
340c906a3a0SJean-Christophe DUBOIS         return;
341c906a3a0SJean-Christophe DUBOIS     }
342c906a3a0SJean-Christophe DUBOIS 
343c906a3a0SJean-Christophe DUBOIS     DPRINTF("reg[%s] <= 0x%" PRIx32 "\n", imx_spi_reg_name(index),
344c906a3a0SJean-Christophe DUBOIS             (uint32_t)value);
345c906a3a0SJean-Christophe DUBOIS 
346fb116b54SPhilippe Mathieu-Daudé     if (!imx_spi_is_enabled(s)) {
347fb116b54SPhilippe Mathieu-Daudé         /* Block is disabled */
348fb116b54SPhilippe Mathieu-Daudé         if (index != ECSPI_CONREG) {
349fb116b54SPhilippe Mathieu-Daudé             /* Ignore access */
350fb116b54SPhilippe Mathieu-Daudé             return;
351fb116b54SPhilippe Mathieu-Daudé         }
352fb116b54SPhilippe Mathieu-Daudé     }
353fb116b54SPhilippe Mathieu-Daudé 
354c906a3a0SJean-Christophe DUBOIS     change_mask = s->regs[index] ^ value;
355c906a3a0SJean-Christophe DUBOIS 
356c906a3a0SJean-Christophe DUBOIS     switch (index) {
357c906a3a0SJean-Christophe DUBOIS     case ECSPI_RXDATA:
358c906a3a0SJean-Christophe DUBOIS         qemu_log_mask(LOG_GUEST_ERROR, "[%s]%s: Trying to write to RX FIFO\n",
359c906a3a0SJean-Christophe DUBOIS                       TYPE_IMX_SPI, __func__);
360c906a3a0SJean-Christophe DUBOIS         break;
361c906a3a0SJean-Christophe DUBOIS     case ECSPI_TXDATA:
362fb116b54SPhilippe Mathieu-Daudé         if (fifo32_is_full(&s->tx_fifo)) {
363c906a3a0SJean-Christophe DUBOIS             /* Ignore writes if queue is full */
364c906a3a0SJean-Christophe DUBOIS             break;
365c906a3a0SJean-Christophe DUBOIS         }
366c906a3a0SJean-Christophe DUBOIS 
367c906a3a0SJean-Christophe DUBOIS         fifo32_push(&s->tx_fifo, (uint32_t)value);
368c906a3a0SJean-Christophe DUBOIS 
369c906a3a0SJean-Christophe DUBOIS         if (imx_spi_channel_is_master(s) &&
370c906a3a0SJean-Christophe DUBOIS             (s->regs[ECSPI_CONREG] & ECSPI_CONREG_SMC)) {
371c906a3a0SJean-Christophe DUBOIS             /*
372c906a3a0SJean-Christophe DUBOIS              * Start emitting if current channel is master and SMC bit is
373c906a3a0SJean-Christophe DUBOIS              * set.
374c906a3a0SJean-Christophe DUBOIS              */
375c906a3a0SJean-Christophe DUBOIS             imx_spi_flush_txfifo(s);
376c906a3a0SJean-Christophe DUBOIS         }
377c906a3a0SJean-Christophe DUBOIS 
378c906a3a0SJean-Christophe DUBOIS         break;
379c906a3a0SJean-Christophe DUBOIS     case ECSPI_STATREG:
380c906a3a0SJean-Christophe DUBOIS         /* the RO and TC bits are write-one-to-clear */
381c906a3a0SJean-Christophe DUBOIS         value &= ECSPI_STATREG_RO | ECSPI_STATREG_TC;
382c906a3a0SJean-Christophe DUBOIS         s->regs[ECSPI_STATREG] &= ~value;
383c906a3a0SJean-Christophe DUBOIS 
384c906a3a0SJean-Christophe DUBOIS         break;
385c906a3a0SJean-Christophe DUBOIS     case ECSPI_CONREG:
386c906a3a0SJean-Christophe DUBOIS         s->regs[ECSPI_CONREG] = value;
387c906a3a0SJean-Christophe DUBOIS 
38824bf8ef3SBin Meng         burst = EXTRACT(s->regs[ECSPI_CONREG], ECSPI_CONREG_BURST_LENGTH) + 1;
38924bf8ef3SBin Meng         if (burst % 8) {
39024bf8ef3SBin Meng             qemu_log_mask(LOG_UNIMP,
39124bf8ef3SBin Meng                           "[%s]%s: burst length %d not supported: rounding up to next multiple of 8\n",
39224bf8ef3SBin Meng                           TYPE_IMX_SPI, __func__, burst);
39324bf8ef3SBin Meng         }
39424bf8ef3SBin Meng 
395c906a3a0SJean-Christophe DUBOIS         if (!imx_spi_is_enabled(s)) {
3963c9829e5SBin Meng             /* device is disabled, so this is a soft reset */
3973c9829e5SBin Meng             imx_spi_soft_reset(s);
3983c9829e5SBin Meng 
399c906a3a0SJean-Christophe DUBOIS             return;
400c906a3a0SJean-Christophe DUBOIS         }
401c906a3a0SJean-Christophe DUBOIS 
402c906a3a0SJean-Christophe DUBOIS         if (imx_spi_channel_is_master(s)) {
403c906a3a0SJean-Christophe DUBOIS             int i;
404c906a3a0SJean-Christophe DUBOIS 
405c906a3a0SJean-Christophe DUBOIS             /* We are in master mode */
406c906a3a0SJean-Christophe DUBOIS 
4071da79eccSBin Meng             for (i = 0; i < ECSPI_NUM_CS; i++) {
408c906a3a0SJean-Christophe DUBOIS                 qemu_set_irq(s->cs_lines[i],
409c906a3a0SJean-Christophe DUBOIS                              i == imx_spi_selected_channel(s) ? 0 : 1);
410c906a3a0SJean-Christophe DUBOIS             }
411c906a3a0SJean-Christophe DUBOIS 
412c906a3a0SJean-Christophe DUBOIS             if ((value & change_mask & ECSPI_CONREG_SMC) &&
413c906a3a0SJean-Christophe DUBOIS                 !fifo32_is_empty(&s->tx_fifo)) {
414c906a3a0SJean-Christophe DUBOIS                 /* SMC bit is set and TX FIFO has some slots filled in */
415c906a3a0SJean-Christophe DUBOIS                 imx_spi_flush_txfifo(s);
416c906a3a0SJean-Christophe DUBOIS             } else if ((value & change_mask & ECSPI_CONREG_XCH) &&
417c906a3a0SJean-Christophe DUBOIS                 !(value & ECSPI_CONREG_SMC)) {
418c906a3a0SJean-Christophe DUBOIS                 /* This is a request to start emitting */
419c906a3a0SJean-Christophe DUBOIS                 imx_spi_flush_txfifo(s);
420c906a3a0SJean-Christophe DUBOIS             }
421c906a3a0SJean-Christophe DUBOIS         }
422c906a3a0SJean-Christophe DUBOIS 
423c906a3a0SJean-Christophe DUBOIS         break;
424556899fcSJean-Christophe Dubois     case ECSPI_MSGDATA:
425556899fcSJean-Christophe Dubois         /* it is not clear from the spec what MSGDATA is for */
426556899fcSJean-Christophe Dubois         /* Anyway it is not used by Linux driver */
427556899fcSJean-Christophe Dubois         /* So for now we just ignore it */
428556899fcSJean-Christophe Dubois         qemu_log_mask(LOG_UNIMP,
429556899fcSJean-Christophe Dubois                       "[%s]%s: Trying to write to MSGDATA, ignoring\n",
430556899fcSJean-Christophe Dubois                       TYPE_IMX_SPI, __func__);
431556899fcSJean-Christophe Dubois         break;
432c906a3a0SJean-Christophe DUBOIS     default:
433c906a3a0SJean-Christophe DUBOIS         s->regs[index] = value;
434c906a3a0SJean-Christophe DUBOIS 
435c906a3a0SJean-Christophe DUBOIS         break;
436c906a3a0SJean-Christophe DUBOIS     }
437c906a3a0SJean-Christophe DUBOIS 
438c906a3a0SJean-Christophe DUBOIS     imx_spi_update_irq(s);
439c906a3a0SJean-Christophe DUBOIS }
440c906a3a0SJean-Christophe DUBOIS 
441c906a3a0SJean-Christophe DUBOIS static const struct MemoryRegionOps imx_spi_ops = {
442c906a3a0SJean-Christophe DUBOIS     .read = imx_spi_read,
443c906a3a0SJean-Christophe DUBOIS     .write = imx_spi_write,
444c906a3a0SJean-Christophe DUBOIS     .endianness = DEVICE_NATIVE_ENDIAN,
445c906a3a0SJean-Christophe DUBOIS     .valid = {
446c906a3a0SJean-Christophe DUBOIS         /*
447c906a3a0SJean-Christophe DUBOIS          * Our device would not work correctly if the guest was doing
448c906a3a0SJean-Christophe DUBOIS          * unaligned access. This might not be a limitation on the real
449c906a3a0SJean-Christophe DUBOIS          * device but in practice there is no reason for a guest to access
450c906a3a0SJean-Christophe DUBOIS          * this device unaligned.
451c906a3a0SJean-Christophe DUBOIS          */
452c906a3a0SJean-Christophe DUBOIS         .min_access_size = 4,
453c906a3a0SJean-Christophe DUBOIS         .max_access_size = 4,
454c906a3a0SJean-Christophe DUBOIS         .unaligned = false,
455c906a3a0SJean-Christophe DUBOIS     },
456c906a3a0SJean-Christophe DUBOIS };
457c906a3a0SJean-Christophe DUBOIS 
imx_spi_realize(DeviceState * dev,Error ** errp)458c906a3a0SJean-Christophe DUBOIS static void imx_spi_realize(DeviceState *dev, Error **errp)
459c906a3a0SJean-Christophe DUBOIS {
460c906a3a0SJean-Christophe DUBOIS     IMXSPIState *s = IMX_SPI(dev);
461c906a3a0SJean-Christophe DUBOIS     int i;
462c906a3a0SJean-Christophe DUBOIS 
463c906a3a0SJean-Christophe DUBOIS     s->bus = ssi_create_bus(dev, "spi");
464c906a3a0SJean-Christophe DUBOIS 
465c906a3a0SJean-Christophe DUBOIS     memory_region_init_io(&s->iomem, OBJECT(dev), &imx_spi_ops, s,
466c906a3a0SJean-Christophe DUBOIS                           TYPE_IMX_SPI, 0x1000);
467c906a3a0SJean-Christophe DUBOIS     sysbus_init_mmio(SYS_BUS_DEVICE(dev), &s->iomem);
468c906a3a0SJean-Christophe DUBOIS     sysbus_init_irq(SYS_BUS_DEVICE(dev), &s->irq);
469c906a3a0SJean-Christophe DUBOIS 
4701da79eccSBin Meng     for (i = 0; i < ECSPI_NUM_CS; ++i) {
471c906a3a0SJean-Christophe DUBOIS         sysbus_init_irq(SYS_BUS_DEVICE(dev), &s->cs_lines[i]);
472c906a3a0SJean-Christophe DUBOIS     }
473c906a3a0SJean-Christophe DUBOIS 
474c906a3a0SJean-Christophe DUBOIS     fifo32_create(&s->tx_fifo, ECSPI_FIFO_SIZE);
475c906a3a0SJean-Christophe DUBOIS     fifo32_create(&s->rx_fifo, ECSPI_FIFO_SIZE);
476c906a3a0SJean-Christophe DUBOIS }
477c906a3a0SJean-Christophe DUBOIS 
imx_spi_class_init(ObjectClass * klass,void * data)478c906a3a0SJean-Christophe DUBOIS static void imx_spi_class_init(ObjectClass *klass, void *data)
479c906a3a0SJean-Christophe DUBOIS {
480c906a3a0SJean-Christophe DUBOIS     DeviceClass *dc = DEVICE_CLASS(klass);
481c906a3a0SJean-Christophe DUBOIS 
482c906a3a0SJean-Christophe DUBOIS     dc->realize = imx_spi_realize;
483c906a3a0SJean-Christophe DUBOIS     dc->vmsd = &vmstate_imx_spi;
484*e3d08143SPeter Maydell     device_class_set_legacy_reset(dc, imx_spi_reset);
485c906a3a0SJean-Christophe DUBOIS     dc->desc = "i.MX SPI Controller";
486c906a3a0SJean-Christophe DUBOIS }
487c906a3a0SJean-Christophe DUBOIS 
488c906a3a0SJean-Christophe DUBOIS static const TypeInfo imx_spi_info = {
489c906a3a0SJean-Christophe DUBOIS     .name          = TYPE_IMX_SPI,
490c906a3a0SJean-Christophe DUBOIS     .parent        = TYPE_SYS_BUS_DEVICE,
491c906a3a0SJean-Christophe DUBOIS     .instance_size = sizeof(IMXSPIState),
492c906a3a0SJean-Christophe DUBOIS     .class_init    = imx_spi_class_init,
493c906a3a0SJean-Christophe DUBOIS };
494c906a3a0SJean-Christophe DUBOIS 
imx_spi_register_types(void)495c906a3a0SJean-Christophe DUBOIS static void imx_spi_register_types(void)
496c906a3a0SJean-Christophe DUBOIS {
497c906a3a0SJean-Christophe DUBOIS     type_register_static(&imx_spi_info);
498c906a3a0SJean-Christophe DUBOIS }
499c906a3a0SJean-Christophe DUBOIS 
500c906a3a0SJean-Christophe DUBOIS type_init(imx_spi_register_types)
501