xref: /qemu/hw/char/imx_serial.c (revision 06b40d250ecfa1633209c2e431a7a38acfd03a98)
140b6f911SPeter Chubb /*
240b6f911SPeter Chubb  * IMX31 UARTS
340b6f911SPeter Chubb  *
440b6f911SPeter Chubb  * Copyright (c) 2008 OKL
540b6f911SPeter Chubb  * Originally Written by Hans Jiang
640b6f911SPeter Chubb  * Copyright (c) 2011 NICTA Pty Ltd.
7cd0bda20SJean-Christophe Dubois  * Updated by Jean-Christophe Dubois <jcd@tribudubois.net>
840b6f911SPeter Chubb  *
940b6f911SPeter Chubb  * This work is licensed under the terms of the GNU GPL, version 2 or later.
1040b6f911SPeter Chubb  * See the COPYING file in the top-level directory.
1140b6f911SPeter Chubb  *
1240b6f911SPeter Chubb  * This is a `bare-bones' implementation of the IMX series serial ports.
1340b6f911SPeter Chubb  * TODO:
1440b6f911SPeter Chubb  *  -- implement FIFOs.  The real hardware has 32 word transmit
1540b6f911SPeter Chubb  *                       and receive FIFOs; we currently use a 1-char buffer
1640b6f911SPeter Chubb  *  -- implement DMA
1740b6f911SPeter Chubb  *  -- implement BAUD-rate and modem lines, for when the backend
1840b6f911SPeter Chubb  *     is a real serial device.
1940b6f911SPeter Chubb  */
2040b6f911SPeter Chubb 
218ef94f0bSPeter Maydell #include "qemu/osdep.h"
22cd0bda20SJean-Christophe Dubois #include "hw/char/imx_serial.h"
2364552b6bSMarkus Armbruster #include "hw/irq.h"
24a27bd6c7SMarkus Armbruster #include "hw/qdev-properties.h"
25ce35e229SEduardo Habkost #include "hw/qdev-properties-system.h"
26d6454270SMarkus Armbruster #include "migration/vmstate.h"
2703dd024fSPaolo Bonzini #include "qemu/log.h"
280b8fa32fSMarkus Armbruster #include "qemu/module.h"
29988f2442SRayhan Faizel #include "qemu/fifo32.h"
301bada3c9SBernhard Beschow #include "trace.h"
3140b6f911SPeter Chubb 
328ccce77cSJean-Christophe Dubois #ifndef DEBUG_IMX_UART
338ccce77cSJean-Christophe Dubois #define DEBUG_IMX_UART 0
3440b6f911SPeter Chubb #endif
3540b6f911SPeter Chubb 
368ccce77cSJean-Christophe Dubois #define DPRINTF(fmt, args...) \
378ccce77cSJean-Christophe Dubois     do { \
388ccce77cSJean-Christophe Dubois         if (DEBUG_IMX_UART) { \
398ccce77cSJean-Christophe Dubois             fprintf(stderr, "[%s]%s: " fmt , TYPE_IMX_SERIAL, \
408ccce77cSJean-Christophe Dubois                                              __func__, ##args); \
418ccce77cSJean-Christophe Dubois         } \
428ccce77cSJean-Christophe Dubois     } while (0)
4340b6f911SPeter Chubb 
4440b6f911SPeter Chubb static const VMStateDescription vmstate_imx_serial = {
45fa2650a3SJean-Christophe Dubois     .name = TYPE_IMX_SERIAL,
46988f2442SRayhan Faizel     .version_id = 3,
47988f2442SRayhan Faizel     .minimum_version_id = 3,
482f6cab05SRichard Henderson     .fields = (const VMStateField[]) {
49988f2442SRayhan Faizel         VMSTATE_FIFO32(rx_fifo, IMXSerialState),
50988f2442SRayhan Faizel         VMSTATE_TIMER(ageing_timer, IMXSerialState),
5140b6f911SPeter Chubb         VMSTATE_UINT32(usr1, IMXSerialState),
5240b6f911SPeter Chubb         VMSTATE_UINT32(usr2, IMXSerialState),
5340b6f911SPeter Chubb         VMSTATE_UINT32(ucr1, IMXSerialState),
5440b6f911SPeter Chubb         VMSTATE_UINT32(uts1, IMXSerialState),
5540b6f911SPeter Chubb         VMSTATE_UINT32(onems, IMXSerialState),
5640b6f911SPeter Chubb         VMSTATE_UINT32(ufcr, IMXSerialState),
5740b6f911SPeter Chubb         VMSTATE_UINT32(ubmr, IMXSerialState),
5840b6f911SPeter Chubb         VMSTATE_UINT32(ubrc, IMXSerialState),
5940b6f911SPeter Chubb         VMSTATE_UINT32(ucr3, IMXSerialState),
6046d3fb63SAndrey Smirnov         VMSTATE_UINT32(ucr4, IMXSerialState),
6140b6f911SPeter Chubb         VMSTATE_END_OF_LIST()
6240b6f911SPeter Chubb     },
6340b6f911SPeter Chubb };
6440b6f911SPeter Chubb 
imx_update(IMXSerialState * s)6540b6f911SPeter Chubb static void imx_update(IMXSerialState *s)
6640b6f911SPeter Chubb {
67824e4a12SAndrey Smirnov     uint32_t usr1;
68824e4a12SAndrey Smirnov     uint32_t usr2;
69824e4a12SAndrey Smirnov     uint32_t mask;
7040b6f911SPeter Chubb 
71824e4a12SAndrey Smirnov     /*
72824e4a12SAndrey Smirnov      * Lucky for us TRDY and RRDY has the same offset in both USR1 and
73824e4a12SAndrey Smirnov      * UCR1, so we can get away with something as simple as the
74824e4a12SAndrey Smirnov      * following:
75824e4a12SAndrey Smirnov      */
76824e4a12SAndrey Smirnov     usr1 = s->usr1 & s->ucr1 & (USR1_TRDY | USR1_RRDY);
77824e4a12SAndrey Smirnov     /*
78988f2442SRayhan Faizel      * Interrupt if AGTIM is set (ageing timer interrupt in RxFIFO)
79988f2442SRayhan Faizel      */
80988f2442SRayhan Faizel     usr1 |= (s->ucr2 & UCR2_ATEN) ? (s->usr1 & USR1_AGTIM) : 0;
81988f2442SRayhan Faizel     /*
82824e4a12SAndrey Smirnov      * Bits that we want in USR2 are not as conveniently laid out,
83824e4a12SAndrey Smirnov      * unfortunately.
84824e4a12SAndrey Smirnov      */
85824e4a12SAndrey Smirnov     mask = (s->ucr1 & UCR1_TXMPTYEN) ? USR2_TXFE : 0;
8646d3fb63SAndrey Smirnov     /*
8746d3fb63SAndrey Smirnov      * TCEN and TXDC are both bit 3
88988f2442SRayhan Faizel      * ORE and OREN are both bit 1
893c54cf77SHans-Erik Floryd      * RDR and DREN are both bit 0
9046d3fb63SAndrey Smirnov      */
91988f2442SRayhan Faizel     mask |= s->ucr4 & (UCR4_WKEN | UCR4_TCEN | UCR4_DREN | UCR4_OREN);
9246d3fb63SAndrey Smirnov 
93824e4a12SAndrey Smirnov     usr2 = s->usr2 & mask;
9440b6f911SPeter Chubb 
95824e4a12SAndrey Smirnov     qemu_set_irq(s->irq, usr1 || usr2);
9640b6f911SPeter Chubb }
9740b6f911SPeter Chubb 
imx_serial_rx_fifo_push(IMXSerialState * s,uint32_t value)98988f2442SRayhan Faizel static void imx_serial_rx_fifo_push(IMXSerialState *s, uint32_t value)
99988f2442SRayhan Faizel {
100988f2442SRayhan Faizel     uint32_t pushed_value = value;
101988f2442SRayhan Faizel     if (fifo32_is_full(&s->rx_fifo)) {
102988f2442SRayhan Faizel         /* Set ORE if FIFO is already full */
103988f2442SRayhan Faizel         s->usr2 |= USR2_ORE;
104988f2442SRayhan Faizel     } else {
105988f2442SRayhan Faizel         if (fifo32_num_used(&s->rx_fifo) == FIFO_SIZE - 1) {
106988f2442SRayhan Faizel             /* Set OVRRUN on 32nd character in FIFO */
107988f2442SRayhan Faizel             pushed_value |= URXD_ERR | URXD_OVRRUN;
108988f2442SRayhan Faizel         }
109988f2442SRayhan Faizel         fifo32_push(&s->rx_fifo, pushed_value);
110988f2442SRayhan Faizel     }
111988f2442SRayhan Faizel }
112988f2442SRayhan Faizel 
imx_serial_rx_fifo_pop(IMXSerialState * s)113988f2442SRayhan Faizel static uint32_t imx_serial_rx_fifo_pop(IMXSerialState *s)
114988f2442SRayhan Faizel {
115988f2442SRayhan Faizel     if (fifo32_is_empty(&s->rx_fifo)) {
116988f2442SRayhan Faizel         return 0;
117988f2442SRayhan Faizel     }
118988f2442SRayhan Faizel     return fifo32_pop(&s->rx_fifo);
119988f2442SRayhan Faizel }
120988f2442SRayhan Faizel 
imx_serial_rx_fifo_ageing_timer_int(void * opaque)121988f2442SRayhan Faizel static void imx_serial_rx_fifo_ageing_timer_int(void *opaque)
122988f2442SRayhan Faizel {
123988f2442SRayhan Faizel     IMXSerialState *s = (IMXSerialState *) opaque;
124988f2442SRayhan Faizel     s->usr1 |= USR1_AGTIM;
125988f2442SRayhan Faizel     imx_update(s);
126988f2442SRayhan Faizel }
127988f2442SRayhan Faizel 
imx_serial_rx_fifo_ageing_timer_restart(void * opaque)128988f2442SRayhan Faizel static void imx_serial_rx_fifo_ageing_timer_restart(void *opaque)
129988f2442SRayhan Faizel {
130988f2442SRayhan Faizel     /*
131988f2442SRayhan Faizel      * Ageing timer starts ticking when
132988f2442SRayhan Faizel      * RX FIFO is non empty and below trigger level.
133988f2442SRayhan Faizel      * Timer is reset if new character is received or
134988f2442SRayhan Faizel      * a FIFO read occurs.
135988f2442SRayhan Faizel      * Timer triggers an interrupt when duration of
136988f2442SRayhan Faizel      * 8 characters has passed (assuming 115200 baudrate).
137988f2442SRayhan Faizel      */
138988f2442SRayhan Faizel     IMXSerialState *s = (IMXSerialState *) opaque;
139988f2442SRayhan Faizel 
140988f2442SRayhan Faizel     if (!(s->usr1 & USR1_RRDY) && !(s->uts1 & UTS1_RXEMPTY)) {
141988f2442SRayhan Faizel         timer_mod_ns(&s->ageing_timer,
142988f2442SRayhan Faizel                      qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL) + AGE_DURATION_NS);
143988f2442SRayhan Faizel     } else {
144988f2442SRayhan Faizel         timer_del(&s->ageing_timer);
145988f2442SRayhan Faizel     }
146988f2442SRayhan Faizel }
147988f2442SRayhan Faizel 
imx_serial_reset(IMXSerialState * s)14840b6f911SPeter Chubb static void imx_serial_reset(IMXSerialState *s)
14940b6f911SPeter Chubb {
15040b6f911SPeter Chubb 
15140b6f911SPeter Chubb     s->usr1 = USR1_TRDY | USR1_RXDS;
15240b6f911SPeter Chubb     /*
15340b6f911SPeter Chubb      * Fake attachment of a terminal: assert RTS.
15440b6f911SPeter Chubb      */
15540b6f911SPeter Chubb     s->usr1 |= USR1_RTSS;
15640b6f911SPeter Chubb     s->usr2 = USR2_TXFE | USR2_TXDC | USR2_DCDIN;
15740b6f911SPeter Chubb     s->uts1 = UTS1_RXEMPTY | UTS1_TXEMPTY;
15840b6f911SPeter Chubb     s->ucr1 = 0;
15940b6f911SPeter Chubb     s->ucr2 = UCR2_SRST;
16040b6f911SPeter Chubb     s->ucr3 = 0x700;
16140b6f911SPeter Chubb     s->ubmr = 0;
16240b6f911SPeter Chubb     s->ubrc = 4;
163b6cd77fbSBernhard Beschow     s->ufcr = BIT(11) | BIT(0);
164988f2442SRayhan Faizel 
165988f2442SRayhan Faizel     fifo32_reset(&s->rx_fifo);
166988f2442SRayhan Faizel     timer_del(&s->ageing_timer);
16740b6f911SPeter Chubb }
16840b6f911SPeter Chubb 
imx_serial_reset_at_boot(DeviceState * dev)16940b6f911SPeter Chubb static void imx_serial_reset_at_boot(DeviceState *dev)
17040b6f911SPeter Chubb {
1718d8e3481SAndreas Färber     IMXSerialState *s = IMX_SERIAL(dev);
17240b6f911SPeter Chubb 
17340b6f911SPeter Chubb     imx_serial_reset(s);
17440b6f911SPeter Chubb 
17540b6f911SPeter Chubb     /*
1769b4b4e51SMichael Tokarev      * enable the uart on boot, so messages from the linux decompressor
17740b6f911SPeter Chubb      * are visible.  On real hardware this is done by the boot rom
17840b6f911SPeter Chubb      * before anything else is loaded.
17940b6f911SPeter Chubb      */
18040b6f911SPeter Chubb     s->ucr1 = UCR1_UARTEN;
18140b6f911SPeter Chubb     s->ucr2 = UCR2_TXEN;
18240b6f911SPeter Chubb 
18340b6f911SPeter Chubb }
18440b6f911SPeter Chubb 
imx_serial_read(void * opaque,hwaddr offset,unsigned size)185a8170e5eSAvi Kivity static uint64_t imx_serial_read(void *opaque, hwaddr offset,
18640b6f911SPeter Chubb                                 unsigned size)
18740b6f911SPeter Chubb {
18840b6f911SPeter Chubb     IMXSerialState *s = (IMXSerialState *)opaque;
1891bada3c9SBernhard Beschow     Chardev *chr = qemu_chr_fe_get_driver(&s->chr);
190988f2442SRayhan Faizel     uint32_t c, rx_used;
191988f2442SRayhan Faizel     uint8_t rxtl = s->ufcr & TL_MASK;
1921bada3c9SBernhard Beschow     uint64_t value;
1938ccce77cSJean-Christophe Dubois 
19440b6f911SPeter Chubb     switch (offset >> 2) {
19540b6f911SPeter Chubb     case 0x0: /* URXD */
196988f2442SRayhan Faizel         c = imx_serial_rx_fifo_pop(s);
19740b6f911SPeter Chubb         if (!(s->uts1 & UTS1_RXEMPTY)) {
19840b6f911SPeter Chubb             /* Character is valid */
19940b6f911SPeter Chubb             c |= URXD_CHARRDY;
200988f2442SRayhan Faizel             rx_used = fifo32_num_used(&s->rx_fifo);
201988f2442SRayhan Faizel             /* Clear RRDY if below threshold */
202988f2442SRayhan Faizel             if (rx_used < rxtl) {
20340b6f911SPeter Chubb                 s->usr1 &= ~USR1_RRDY;
204988f2442SRayhan Faizel             }
205988f2442SRayhan Faizel             if (rx_used == 0) {
20640b6f911SPeter Chubb                 s->usr2 &= ~USR2_RDR;
20740b6f911SPeter Chubb                 s->uts1 |= UTS1_RXEMPTY;
208988f2442SRayhan Faizel             }
20940b6f911SPeter Chubb             imx_update(s);
210988f2442SRayhan Faizel             imx_serial_rx_fifo_ageing_timer_restart(s);
2115345fdb4SMarc-André Lureau             qemu_chr_fe_accept_input(&s->chr);
21240b6f911SPeter Chubb         }
2131bada3c9SBernhard Beschow         value = c;
2141bada3c9SBernhard Beschow         break;
21540b6f911SPeter Chubb 
21640b6f911SPeter Chubb     case 0x20: /* UCR1 */
2171bada3c9SBernhard Beschow         value = s->ucr1;
2181bada3c9SBernhard Beschow         break;
21940b6f911SPeter Chubb 
22040b6f911SPeter Chubb     case 0x21: /* UCR2 */
2211bada3c9SBernhard Beschow         value = s->ucr2;
2221bada3c9SBernhard Beschow         break;
22340b6f911SPeter Chubb 
22440b6f911SPeter Chubb     case 0x25: /* USR1 */
2251bada3c9SBernhard Beschow         value = s->usr1;
2261bada3c9SBernhard Beschow         break;
22740b6f911SPeter Chubb 
22840b6f911SPeter Chubb     case 0x26: /* USR2 */
2291bada3c9SBernhard Beschow         value = s->usr2;
2301bada3c9SBernhard Beschow         break;
23140b6f911SPeter Chubb 
23240b6f911SPeter Chubb     case 0x2A: /* BRM Modulator */
2331bada3c9SBernhard Beschow         value = s->ubmr;
2341bada3c9SBernhard Beschow         break;
23540b6f911SPeter Chubb 
23640b6f911SPeter Chubb     case 0x2B: /* Baud Rate Count */
2371bada3c9SBernhard Beschow         value = s->ubrc;
2381bada3c9SBernhard Beschow         break;
23940b6f911SPeter Chubb 
24040b6f911SPeter Chubb     case 0x2d: /* Test register */
2411bada3c9SBernhard Beschow         value = s->uts1;
2421bada3c9SBernhard Beschow         break;
24340b6f911SPeter Chubb 
24440b6f911SPeter Chubb     case 0x24: /* UFCR */
2451bada3c9SBernhard Beschow         value = s->ufcr;
2461bada3c9SBernhard Beschow         break;
24740b6f911SPeter Chubb 
24840b6f911SPeter Chubb     case 0x2c:
2491bada3c9SBernhard Beschow         value = s->onems;
2501bada3c9SBernhard Beschow         break;
25140b6f911SPeter Chubb 
25240b6f911SPeter Chubb     case 0x22: /* UCR3 */
2531bada3c9SBernhard Beschow         value = s->ucr3;
2541bada3c9SBernhard Beschow         break;
25540b6f911SPeter Chubb 
25640b6f911SPeter Chubb     case 0x23: /* UCR4 */
2571bada3c9SBernhard Beschow         value = s->ucr4;
2581bada3c9SBernhard Beschow         break;
25946d3fb63SAndrey Smirnov 
26040b6f911SPeter Chubb     case 0x29: /* BRM Incremental */
2611bada3c9SBernhard Beschow         value = 0x0; /* TODO */
2621bada3c9SBernhard Beschow         break;
26340b6f911SPeter Chubb 
26440b6f911SPeter Chubb     default:
2658ccce77cSJean-Christophe Dubois         qemu_log_mask(LOG_GUEST_ERROR, "[%s]%s: Bad register at offset 0x%"
2668ccce77cSJean-Christophe Dubois                       HWADDR_PRIx "\n", TYPE_IMX_SERIAL, __func__, offset);
2671bada3c9SBernhard Beschow         value = 0;
2681bada3c9SBernhard Beschow         break;
26940b6f911SPeter Chubb     }
2701bada3c9SBernhard Beschow 
2711bada3c9SBernhard Beschow     trace_imx_serial_read(chr ? chr->label : "NODEV", offset, value);
2721bada3c9SBernhard Beschow 
2731bada3c9SBernhard Beschow     return value;
27440b6f911SPeter Chubb }
27540b6f911SPeter Chubb 
imx_serial_write(void * opaque,hwaddr offset,uint64_t value,unsigned size)276a8170e5eSAvi Kivity static void imx_serial_write(void *opaque, hwaddr offset,
27740b6f911SPeter Chubb                              uint64_t value, unsigned size)
27840b6f911SPeter Chubb {
27940b6f911SPeter Chubb     IMXSerialState *s = (IMXSerialState *)opaque;
2800ec7b3e7SMarc-André Lureau     Chardev *chr = qemu_chr_fe_get_driver(&s->chr);
28140b6f911SPeter Chubb     unsigned char ch;
28240b6f911SPeter Chubb 
2831bada3c9SBernhard Beschow     trace_imx_serial_write(chr ? chr->label : "NODEV", offset, value);
28440b6f911SPeter Chubb 
28540b6f911SPeter Chubb     switch (offset >> 2) {
28640b6f911SPeter Chubb     case 0x10: /* UTXD */
28740b6f911SPeter Chubb         ch = value;
28840b6f911SPeter Chubb         if (s->ucr2 & UCR2_TXEN) {
2896ab3fc32SDaniel P. Berrange             /* XXX this blocks entire thread. Rewrite to use
2906ab3fc32SDaniel P. Berrange              * qemu_chr_fe_write and background I/O callbacks */
2915345fdb4SMarc-André Lureau             qemu_chr_fe_write_all(&s->chr, &ch, 1);
29240b6f911SPeter Chubb             s->usr1 &= ~USR1_TRDY;
29346d3fb63SAndrey Smirnov             s->usr2 &= ~USR2_TXDC;
29440b6f911SPeter Chubb             imx_update(s);
29540b6f911SPeter Chubb             s->usr1 |= USR1_TRDY;
29646d3fb63SAndrey Smirnov             s->usr2 |= USR2_TXDC;
29740b6f911SPeter Chubb             imx_update(s);
29840b6f911SPeter Chubb         }
29940b6f911SPeter Chubb         break;
30040b6f911SPeter Chubb 
30140b6f911SPeter Chubb     case 0x20: /* UCR1 */
30240b6f911SPeter Chubb         s->ucr1 = value & 0xffff;
3038ccce77cSJean-Christophe Dubois 
30440b6f911SPeter Chubb         DPRINTF("write(ucr1=%x)\n", (unsigned int)value);
3058ccce77cSJean-Christophe Dubois 
30640b6f911SPeter Chubb         imx_update(s);
30740b6f911SPeter Chubb         break;
30840b6f911SPeter Chubb 
30940b6f911SPeter Chubb     case 0x21: /* UCR2 */
31040b6f911SPeter Chubb         /*
31140b6f911SPeter Chubb          * Only a few bits in control register 2 are implemented as yet.
31240b6f911SPeter Chubb          * If it's intended to use a real serial device as a back-end, this
31340b6f911SPeter Chubb          * register will have to be implemented more fully.
31440b6f911SPeter Chubb          */
31540b6f911SPeter Chubb         if (!(value & UCR2_SRST)) {
31640b6f911SPeter Chubb             imx_serial_reset(s);
31740b6f911SPeter Chubb             imx_update(s);
31840b6f911SPeter Chubb             value |= UCR2_SRST;
31940b6f911SPeter Chubb         }
32040b6f911SPeter Chubb         if (value & UCR2_RXEN) {
32140b6f911SPeter Chubb             if (!(s->ucr2 & UCR2_RXEN)) {
3225345fdb4SMarc-André Lureau                 qemu_chr_fe_accept_input(&s->chr);
32340b6f911SPeter Chubb             }
32440b6f911SPeter Chubb         }
32540b6f911SPeter Chubb         s->ucr2 = value & 0xffff;
32640b6f911SPeter Chubb         break;
32740b6f911SPeter Chubb 
32840b6f911SPeter Chubb     case 0x25: /* USR1 */
32940b6f911SPeter Chubb         value &= USR1_AWAKE | USR1_AIRINT | USR1_DTRD | USR1_AGTIM |
33040b6f911SPeter Chubb                  USR1_FRAMERR | USR1_ESCF | USR1_RTSD | USR1_PARTYER;
33140b6f911SPeter Chubb         s->usr1 &= ~value;
33240b6f911SPeter Chubb         break;
33340b6f911SPeter Chubb 
33440b6f911SPeter Chubb     case 0x26: /* USR2 */
33540b6f911SPeter Chubb         /*
33640b6f911SPeter Chubb          * Writing 1 to some bits clears them; all other
33740b6f911SPeter Chubb          * values are ignored
33840b6f911SPeter Chubb          */
33940b6f911SPeter Chubb         value &= USR2_ADET | USR2_DTRF | USR2_IDLE | USR2_ACST |
34040b6f911SPeter Chubb                  USR2_RIDELT | USR2_IRINT | USR2_WAKE |
34140b6f911SPeter Chubb                  USR2_DCDDELT | USR2_RTSF | USR2_BRCD | USR2_ORE;
34240b6f911SPeter Chubb         s->usr2 &= ~value;
34340b6f911SPeter Chubb         break;
34440b6f911SPeter Chubb 
34540b6f911SPeter Chubb     /*
34640b6f911SPeter Chubb      * Linux expects to see what it writes to these registers
34740b6f911SPeter Chubb      * We don't currently alter the baud rate
34840b6f911SPeter Chubb      */
34940b6f911SPeter Chubb     case 0x29: /* UBIR */
35040b6f911SPeter Chubb         s->ubrc = value & 0xffff;
35140b6f911SPeter Chubb         break;
35240b6f911SPeter Chubb 
35340b6f911SPeter Chubb     case 0x2a: /* UBMR */
35440b6f911SPeter Chubb         s->ubmr = value & 0xffff;
35540b6f911SPeter Chubb         break;
35640b6f911SPeter Chubb 
35740b6f911SPeter Chubb     case 0x2c: /* One ms reg */
35840b6f911SPeter Chubb         s->onems = value & 0xffff;
35940b6f911SPeter Chubb         break;
36040b6f911SPeter Chubb 
36140b6f911SPeter Chubb     case 0x24: /* FIFO control register */
36240b6f911SPeter Chubb         s->ufcr = value & 0xffff;
36340b6f911SPeter Chubb         break;
36440b6f911SPeter Chubb 
36540b6f911SPeter Chubb     case 0x22: /* UCR3 */
36640b6f911SPeter Chubb         s->ucr3 = value & 0xffff;
36740b6f911SPeter Chubb         break;
36840b6f911SPeter Chubb 
36940b6f911SPeter Chubb     case 0x23: /* UCR4 */
37046d3fb63SAndrey Smirnov         s->ucr4 = value & 0xffff;
37146d3fb63SAndrey Smirnov         imx_update(s);
37246d3fb63SAndrey Smirnov         break;
37346d3fb63SAndrey Smirnov 
37446d3fb63SAndrey Smirnov     case 0x2d: /* UTS1 */
3758ccce77cSJean-Christophe Dubois         qemu_log_mask(LOG_UNIMP, "[%s]%s: Unimplemented reg 0x%"
3768ccce77cSJean-Christophe Dubois                       HWADDR_PRIx "\n", TYPE_IMX_SERIAL, __func__, offset);
37740b6f911SPeter Chubb         /* TODO */
37840b6f911SPeter Chubb         break;
37940b6f911SPeter Chubb 
38040b6f911SPeter Chubb     default:
3818ccce77cSJean-Christophe Dubois         qemu_log_mask(LOG_GUEST_ERROR, "[%s]%s: Bad register at offset 0x%"
3828ccce77cSJean-Christophe Dubois                       HWADDR_PRIx "\n", TYPE_IMX_SERIAL, __func__, offset);
38340b6f911SPeter Chubb     }
38440b6f911SPeter Chubb }
38540b6f911SPeter Chubb 
imx_can_receive(void * opaque)38640b6f911SPeter Chubb static int imx_can_receive(void *opaque)
38740b6f911SPeter Chubb {
38840b6f911SPeter Chubb     IMXSerialState *s = (IMXSerialState *)opaque;
38991f8c04dSPhilippe Mathieu-Daudé 
39091f8c04dSPhilippe Mathieu-Daudé     return s->ucr2 & UCR2_RXEN ? fifo32_num_free(&s->rx_fifo) : 0;
39140b6f911SPeter Chubb }
39240b6f911SPeter Chubb 
imx_put_data(void * opaque,uint32_t value)39340b6f911SPeter Chubb static void imx_put_data(void *opaque, uint32_t value)
39440b6f911SPeter Chubb {
39540b6f911SPeter Chubb     IMXSerialState *s = (IMXSerialState *)opaque;
3961bada3c9SBernhard Beschow     Chardev *chr = qemu_chr_fe_get_driver(&s->chr);
397988f2442SRayhan Faizel     uint8_t rxtl = s->ufcr & TL_MASK;
3988ccce77cSJean-Christophe Dubois 
3991bada3c9SBernhard Beschow     trace_imx_serial_put_data(chr ? chr->label : "NODEV", value);
4001bada3c9SBernhard Beschow 
401988f2442SRayhan Faizel     imx_serial_rx_fifo_push(s, value);
402988f2442SRayhan Faizel     if (fifo32_num_used(&s->rx_fifo) >= rxtl) {
40340b6f911SPeter Chubb         s->usr1 |= USR1_RRDY;
404988f2442SRayhan Faizel     }
40540b6f911SPeter Chubb     s->usr2 |= USR2_RDR;
40640b6f911SPeter Chubb     s->uts1 &= ~UTS1_RXEMPTY;
407478a573aSTrent Piepho     if (value & URXD_BRK) {
408478a573aSTrent Piepho         s->usr2 |= USR2_BRCD;
409478a573aSTrent Piepho     }
410a451cc11SBernhard Beschow 
411a451cc11SBernhard Beschow     imx_serial_rx_fifo_ageing_timer_restart(s);
412a451cc11SBernhard Beschow 
41340b6f911SPeter Chubb     imx_update(s);
41440b6f911SPeter Chubb }
41540b6f911SPeter Chubb 
imx_receive(void * opaque,const uint8_t * buf,int size)41640b6f911SPeter Chubb static void imx_receive(void *opaque, const uint8_t *buf, int size)
41740b6f911SPeter Chubb {
418bd96e100SMartin Kaiser     IMXSerialState *s = (IMXSerialState *)opaque;
419bd96e100SMartin Kaiser 
420bd96e100SMartin Kaiser     s->usr2 |= USR2_WAKE;
42191f8c04dSPhilippe Mathieu-Daudé 
42291f8c04dSPhilippe Mathieu-Daudé     for (int i = 0; i < size; i++) {
42391f8c04dSPhilippe Mathieu-Daudé         imx_put_data(opaque, buf[i]);
42491f8c04dSPhilippe Mathieu-Daudé     }
42540b6f911SPeter Chubb }
42640b6f911SPeter Chubb 
imx_event(void * opaque,QEMUChrEvent event)427083b266fSPhilippe Mathieu-Daudé static void imx_event(void *opaque, QEMUChrEvent event)
42840b6f911SPeter Chubb {
42940b6f911SPeter Chubb     if (event == CHR_EVENT_BREAK) {
430478a573aSTrent Piepho         imx_put_data(opaque, URXD_BRK | URXD_FRMERR | URXD_ERR);
43140b6f911SPeter Chubb     }
43240b6f911SPeter Chubb }
43340b6f911SPeter Chubb 
43440b6f911SPeter Chubb 
43540b6f911SPeter Chubb static const struct MemoryRegionOps imx_serial_ops = {
43640b6f911SPeter Chubb     .read = imx_serial_read,
43740b6f911SPeter Chubb     .write = imx_serial_write,
43840b6f911SPeter Chubb     .endianness = DEVICE_NATIVE_ENDIAN,
43940b6f911SPeter Chubb };
44040b6f911SPeter Chubb 
imx_serial_realize(DeviceState * dev,Error ** errp)441f6c64000SJean-Christophe Dubois static void imx_serial_realize(DeviceState *dev, Error **errp)
44240b6f911SPeter Chubb {
4438d8e3481SAndreas Färber     IMXSerialState *s = IMX_SERIAL(dev);
44440b6f911SPeter Chubb 
445988f2442SRayhan Faizel     fifo32_create(&s->rx_fifo, FIFO_SIZE);
446988f2442SRayhan Faizel     timer_init_ns(&s->ageing_timer, QEMU_CLOCK_VIRTUAL,
447988f2442SRayhan Faizel                   imx_serial_rx_fifo_ageing_timer_int, s);
448988f2442SRayhan Faizel 
449fa394ed6SMarc-André Lureau     DPRINTF("char dev for uart: %p\n", qemu_chr_fe_get_driver(&s->chr));
450fa394ed6SMarc-André Lureau 
4515345fdb4SMarc-André Lureau     qemu_chr_fe_set_handlers(&s->chr, imx_can_receive, imx_receive,
45281517ba3SAnton Nefedov                              imx_event, NULL, s, NULL, true);
453f6c64000SJean-Christophe Dubois }
45440b6f911SPeter Chubb 
imx_serial_init(Object * obj)455f6c64000SJean-Christophe Dubois static void imx_serial_init(Object *obj)
456f6c64000SJean-Christophe Dubois {
457f6c64000SJean-Christophe Dubois     SysBusDevice *sbd = SYS_BUS_DEVICE(obj);
458f6c64000SJean-Christophe Dubois     IMXSerialState *s = IMX_SERIAL(obj);
459f6c64000SJean-Christophe Dubois 
460f6c64000SJean-Christophe Dubois     memory_region_init_io(&s->iomem, obj, &imx_serial_ops, s,
461f6c64000SJean-Christophe Dubois                           TYPE_IMX_SERIAL, 0x1000);
462f6c64000SJean-Christophe Dubois     sysbus_init_mmio(sbd, &s->iomem);
463f6c64000SJean-Christophe Dubois     sysbus_init_irq(sbd, &s->irq);
46440b6f911SPeter Chubb }
46540b6f911SPeter Chubb 
466312f37d1SRichard Henderson static const Property imx_serial_properties[] = {
46740b6f911SPeter Chubb     DEFINE_PROP_CHR("chardev", IMXSerialState, chr),
46840b6f911SPeter Chubb };
46940b6f911SPeter Chubb 
imx_serial_class_init(ObjectClass * klass,const void * data)470*12d1a768SPhilippe Mathieu-Daudé static void imx_serial_class_init(ObjectClass *klass, const void *data)
47140b6f911SPeter Chubb {
47240b6f911SPeter Chubb     DeviceClass *dc = DEVICE_CLASS(klass);
47340b6f911SPeter Chubb 
474f6c64000SJean-Christophe Dubois     dc->realize = imx_serial_realize;
47540b6f911SPeter Chubb     dc->vmsd = &vmstate_imx_serial;
476e3d08143SPeter Maydell     device_class_set_legacy_reset(dc, imx_serial_reset_at_boot);
477125ee0edSMarcel Apfelbaum     set_bit(DEVICE_CATEGORY_INPUT, dc->categories);
47840b6f911SPeter Chubb     dc->desc = "i.MX series UART";
4794f67d30bSMarc-André Lureau     device_class_set_props(dc, imx_serial_properties);
48040b6f911SPeter Chubb }
48140b6f911SPeter Chubb 
4828c43a6f0SAndreas Färber static const TypeInfo imx_serial_info = {
4838d8e3481SAndreas Färber     .name           = TYPE_IMX_SERIAL,
48440b6f911SPeter Chubb     .parent         = TYPE_SYS_BUS_DEVICE,
48540b6f911SPeter Chubb     .instance_size  = sizeof(IMXSerialState),
486f6c64000SJean-Christophe Dubois     .instance_init  = imx_serial_init,
48740b6f911SPeter Chubb     .class_init     = imx_serial_class_init,
48840b6f911SPeter Chubb };
48940b6f911SPeter Chubb 
imx_serial_register_types(void)49040b6f911SPeter Chubb static void imx_serial_register_types(void)
49140b6f911SPeter Chubb {
49240b6f911SPeter Chubb     type_register_static(&imx_serial_info);
49340b6f911SPeter Chubb }
49440b6f911SPeter Chubb 
49540b6f911SPeter Chubb type_init(imx_serial_register_types)
496