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