Statistics
| Branch: | Revision:

root / hw / xilinx_uartlite.c @ 9a6ee9fd

History | View | Annotate | Download (5.8 kB)

1 ee118d95 Edgar E. Iglesias
/*
2 ee118d95 Edgar E. Iglesias
 * QEMU model of Xilinx uartlite.
3 ee118d95 Edgar E. Iglesias
 *
4 ee118d95 Edgar E. Iglesias
 * Copyright (c) 2009 Edgar E. Iglesias.
5 ee118d95 Edgar E. Iglesias
 *
6 ee118d95 Edgar E. Iglesias
 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 ee118d95 Edgar E. Iglesias
 * of this software and associated documentation files (the "Software"), to deal
8 ee118d95 Edgar E. Iglesias
 * in the Software without restriction, including without limitation the rights
9 ee118d95 Edgar E. Iglesias
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10 ee118d95 Edgar E. Iglesias
 * copies of the Software, and to permit persons to whom the Software is
11 ee118d95 Edgar E. Iglesias
 * furnished to do so, subject to the following conditions:
12 ee118d95 Edgar E. Iglesias
 *
13 ee118d95 Edgar E. Iglesias
 * The above copyright notice and this permission notice shall be included in
14 ee118d95 Edgar E. Iglesias
 * all copies or substantial portions of the Software.
15 ee118d95 Edgar E. Iglesias
 *
16 ee118d95 Edgar E. Iglesias
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 ee118d95 Edgar E. Iglesias
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 ee118d95 Edgar E. Iglesias
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 ee118d95 Edgar E. Iglesias
 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 ee118d95 Edgar E. Iglesias
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 ee118d95 Edgar E. Iglesias
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22 ee118d95 Edgar E. Iglesias
 * THE SOFTWARE.
23 ee118d95 Edgar E. Iglesias
 */
24 ee118d95 Edgar E. Iglesias
25 ee118d95 Edgar E. Iglesias
#include "sysbus.h"
26 927d4878 Paolo Bonzini
#include "char/char.h"
27 ee118d95 Edgar E. Iglesias
28 ee118d95 Edgar E. Iglesias
#define DUART(x)
29 ee118d95 Edgar E. Iglesias
30 ee118d95 Edgar E. Iglesias
#define R_RX            0
31 ee118d95 Edgar E. Iglesias
#define R_TX            1
32 ee118d95 Edgar E. Iglesias
#define R_STATUS        2
33 ee118d95 Edgar E. Iglesias
#define R_CTRL          3
34 ee118d95 Edgar E. Iglesias
#define R_MAX           4
35 ee118d95 Edgar E. Iglesias
36 ee118d95 Edgar E. Iglesias
#define STATUS_RXVALID    0x01
37 ee118d95 Edgar E. Iglesias
#define STATUS_RXFULL     0x02
38 ee118d95 Edgar E. Iglesias
#define STATUS_TXEMPTY    0x04
39 ee118d95 Edgar E. Iglesias
#define STATUS_TXFULL     0x08
40 ee118d95 Edgar E. Iglesias
#define STATUS_IE         0x10
41 ee118d95 Edgar E. Iglesias
#define STATUS_OVERRUN    0x20
42 ee118d95 Edgar E. Iglesias
#define STATUS_FRAME      0x40
43 ee118d95 Edgar E. Iglesias
#define STATUS_PARITY     0x80
44 ee118d95 Edgar E. Iglesias
45 ee118d95 Edgar E. Iglesias
#define CONTROL_RST_TX    0x01
46 ee118d95 Edgar E. Iglesias
#define CONTROL_RST_RX    0x02
47 ee118d95 Edgar E. Iglesias
#define CONTROL_IE        0x10
48 ee118d95 Edgar E. Iglesias
49 ee118d95 Edgar E. Iglesias
struct xlx_uartlite
50 ee118d95 Edgar E. Iglesias
{
51 ee118d95 Edgar E. Iglesias
    SysBusDevice busdev;
52 010f3f5f Edgar E. Iglesias
    MemoryRegion mmio;
53 ee118d95 Edgar E. Iglesias
    CharDriverState *chr;
54 ee118d95 Edgar E. Iglesias
    qemu_irq irq;
55 ee118d95 Edgar E. Iglesias
56 ee118d95 Edgar E. Iglesias
    uint8_t rx_fifo[8];
57 ee118d95 Edgar E. Iglesias
    unsigned int rx_fifo_pos;
58 ee118d95 Edgar E. Iglesias
    unsigned int rx_fifo_len;
59 ee118d95 Edgar E. Iglesias
60 ee118d95 Edgar E. Iglesias
    uint32_t regs[R_MAX];
61 ee118d95 Edgar E. Iglesias
};
62 ee118d95 Edgar E. Iglesias
63 ee118d95 Edgar E. Iglesias
static void uart_update_irq(struct xlx_uartlite *s)
64 ee118d95 Edgar E. Iglesias
{
65 ee118d95 Edgar E. Iglesias
    unsigned int irq;
66 ee118d95 Edgar E. Iglesias
67 ee118d95 Edgar E. Iglesias
    if (s->rx_fifo_len)
68 ee118d95 Edgar E. Iglesias
        s->regs[R_STATUS] |= STATUS_IE;
69 ee118d95 Edgar E. Iglesias
70 ee118d95 Edgar E. Iglesias
    irq = (s->regs[R_STATUS] & STATUS_IE) && (s->regs[R_CTRL] & CONTROL_IE);
71 ee118d95 Edgar E. Iglesias
    qemu_set_irq(s->irq, irq);
72 ee118d95 Edgar E. Iglesias
}
73 ee118d95 Edgar E. Iglesias
74 ee118d95 Edgar E. Iglesias
static void uart_update_status(struct xlx_uartlite *s)
75 ee118d95 Edgar E. Iglesias
{
76 ee118d95 Edgar E. Iglesias
    uint32_t r;
77 ee118d95 Edgar E. Iglesias
78 ee118d95 Edgar E. Iglesias
    r = s->regs[R_STATUS];
79 ee118d95 Edgar E. Iglesias
    r &= ~7;
80 ee118d95 Edgar E. Iglesias
    r |= 1 << 2; /* Tx fifo is always empty. We are fast :) */
81 ee118d95 Edgar E. Iglesias
    r |= (s->rx_fifo_len == sizeof (s->rx_fifo)) << 1;
82 ee118d95 Edgar E. Iglesias
    r |= (!!s->rx_fifo_len);
83 ee118d95 Edgar E. Iglesias
    s->regs[R_STATUS] = r;
84 ee118d95 Edgar E. Iglesias
}
85 ee118d95 Edgar E. Iglesias
86 010f3f5f Edgar E. Iglesias
static uint64_t
87 a8170e5e Avi Kivity
uart_read(void *opaque, hwaddr addr, unsigned int size)
88 ee118d95 Edgar E. Iglesias
{
89 ee118d95 Edgar E. Iglesias
    struct xlx_uartlite *s = opaque;
90 ee118d95 Edgar E. Iglesias
    uint32_t r = 0;
91 ee118d95 Edgar E. Iglesias
    addr >>= 2;
92 ee118d95 Edgar E. Iglesias
    switch (addr)
93 ee118d95 Edgar E. Iglesias
    {
94 ee118d95 Edgar E. Iglesias
        case R_RX:
95 ee118d95 Edgar E. Iglesias
            r = s->rx_fifo[(s->rx_fifo_pos - s->rx_fifo_len) & 7];
96 ee118d95 Edgar E. Iglesias
            if (s->rx_fifo_len)
97 ee118d95 Edgar E. Iglesias
                s->rx_fifo_len--;
98 ee118d95 Edgar E. Iglesias
            uart_update_status(s);
99 ee118d95 Edgar E. Iglesias
            uart_update_irq(s);
100 80625b97 Peter Crosthwaite
            qemu_chr_accept_input(s->chr);
101 ee118d95 Edgar E. Iglesias
            break;
102 ee118d95 Edgar E. Iglesias
103 ee118d95 Edgar E. Iglesias
        default:
104 ee118d95 Edgar E. Iglesias
            if (addr < ARRAY_SIZE(s->regs))
105 ee118d95 Edgar E. Iglesias
                r = s->regs[addr];
106 ee118d95 Edgar E. Iglesias
            DUART(qemu_log("%s addr=%x v=%x\n", __func__, addr, r));
107 ee118d95 Edgar E. Iglesias
            break;
108 ee118d95 Edgar E. Iglesias
    }
109 ee118d95 Edgar E. Iglesias
    return r;
110 ee118d95 Edgar E. Iglesias
}
111 ee118d95 Edgar E. Iglesias
112 ee118d95 Edgar E. Iglesias
static void
113 a8170e5e Avi Kivity
uart_write(void *opaque, hwaddr addr,
114 010f3f5f Edgar E. Iglesias
           uint64_t val64, unsigned int size)
115 ee118d95 Edgar E. Iglesias
{
116 ee118d95 Edgar E. Iglesias
    struct xlx_uartlite *s = opaque;
117 010f3f5f Edgar E. Iglesias
    uint32_t value = val64;
118 ee118d95 Edgar E. Iglesias
    unsigned char ch = value;
119 ee118d95 Edgar E. Iglesias
120 ee118d95 Edgar E. Iglesias
    addr >>= 2;
121 ee118d95 Edgar E. Iglesias
    switch (addr)
122 ee118d95 Edgar E. Iglesias
    {
123 ee118d95 Edgar E. Iglesias
        case R_STATUS:
124 ee118d95 Edgar E. Iglesias
            hw_error("write to UART STATUS?\n");
125 ee118d95 Edgar E. Iglesias
            break;
126 ee118d95 Edgar E. Iglesias
127 ee118d95 Edgar E. Iglesias
        case R_CTRL:
128 ee118d95 Edgar E. Iglesias
            if (value & CONTROL_RST_RX) {
129 ee118d95 Edgar E. Iglesias
                s->rx_fifo_pos = 0;
130 ee118d95 Edgar E. Iglesias
                s->rx_fifo_len = 0;
131 ee118d95 Edgar E. Iglesias
            }
132 ee118d95 Edgar E. Iglesias
            s->regs[addr] = value;
133 ee118d95 Edgar E. Iglesias
            break;
134 ee118d95 Edgar E. Iglesias
135 ee118d95 Edgar E. Iglesias
        case R_TX:
136 ee118d95 Edgar E. Iglesias
            if (s->chr)
137 2cc6e0a1 Anthony Liguori
                qemu_chr_fe_write(s->chr, &ch, 1);
138 ee118d95 Edgar E. Iglesias
139 ee118d95 Edgar E. Iglesias
            s->regs[addr] = value;
140 ee118d95 Edgar E. Iglesias
141 ee118d95 Edgar E. Iglesias
            /* hax.  */
142 ee118d95 Edgar E. Iglesias
            s->regs[R_STATUS] |= STATUS_IE;
143 ee118d95 Edgar E. Iglesias
            break;
144 ee118d95 Edgar E. Iglesias
145 ee118d95 Edgar E. Iglesias
        default:
146 ee118d95 Edgar E. Iglesias
            DUART(printf("%s addr=%x v=%x\n", __func__, addr, value));
147 ee118d95 Edgar E. Iglesias
            if (addr < ARRAY_SIZE(s->regs))
148 ee118d95 Edgar E. Iglesias
                s->regs[addr] = value;
149 ee118d95 Edgar E. Iglesias
            break;
150 ee118d95 Edgar E. Iglesias
    }
151 ee118d95 Edgar E. Iglesias
    uart_update_status(s);
152 ee118d95 Edgar E. Iglesias
    uart_update_irq(s);
153 ee118d95 Edgar E. Iglesias
}
154 ee118d95 Edgar E. Iglesias
155 010f3f5f Edgar E. Iglesias
static const MemoryRegionOps uart_ops = {
156 010f3f5f Edgar E. Iglesias
    .read = uart_read,
157 010f3f5f Edgar E. Iglesias
    .write = uart_write,
158 010f3f5f Edgar E. Iglesias
    .endianness = DEVICE_NATIVE_ENDIAN,
159 010f3f5f Edgar E. Iglesias
    .valid = {
160 010f3f5f Edgar E. Iglesias
        .min_access_size = 1,
161 010f3f5f Edgar E. Iglesias
        .max_access_size = 4
162 010f3f5f Edgar E. Iglesias
    }
163 ee118d95 Edgar E. Iglesias
};
164 ee118d95 Edgar E. Iglesias
165 ee118d95 Edgar E. Iglesias
static void uart_rx(void *opaque, const uint8_t *buf, int size)
166 ee118d95 Edgar E. Iglesias
{
167 ee118d95 Edgar E. Iglesias
    struct xlx_uartlite *s = opaque;
168 ee118d95 Edgar E. Iglesias
169 ee118d95 Edgar E. Iglesias
    /* Got a byte.  */
170 ee118d95 Edgar E. Iglesias
    if (s->rx_fifo_len >= 8) {
171 ee118d95 Edgar E. Iglesias
        printf("WARNING: UART dropped char.\n");
172 ee118d95 Edgar E. Iglesias
        return;
173 ee118d95 Edgar E. Iglesias
    }
174 ee118d95 Edgar E. Iglesias
    s->rx_fifo[s->rx_fifo_pos] = *buf;
175 ee118d95 Edgar E. Iglesias
    s->rx_fifo_pos++;
176 ee118d95 Edgar E. Iglesias
    s->rx_fifo_pos &= 0x7;
177 ee118d95 Edgar E. Iglesias
    s->rx_fifo_len++;
178 ee118d95 Edgar E. Iglesias
179 ee118d95 Edgar E. Iglesias
    uart_update_status(s);
180 ee118d95 Edgar E. Iglesias
    uart_update_irq(s);
181 ee118d95 Edgar E. Iglesias
}
182 ee118d95 Edgar E. Iglesias
183 ee118d95 Edgar E. Iglesias
static int uart_can_rx(void *opaque)
184 ee118d95 Edgar E. Iglesias
{
185 ee118d95 Edgar E. Iglesias
    struct xlx_uartlite *s = opaque;
186 ee118d95 Edgar E. Iglesias
187 859cc10d Peter Crosthwaite
    return s->rx_fifo_len < sizeof(s->rx_fifo);
188 ee118d95 Edgar E. Iglesias
}
189 ee118d95 Edgar E. Iglesias
190 ee118d95 Edgar E. Iglesias
static void uart_event(void *opaque, int event)
191 ee118d95 Edgar E. Iglesias
{
192 ee118d95 Edgar E. Iglesias
193 ee118d95 Edgar E. Iglesias
}
194 ee118d95 Edgar E. Iglesias
195 81a322d4 Gerd Hoffmann
static int xilinx_uartlite_init(SysBusDevice *dev)
196 ee118d95 Edgar E. Iglesias
{
197 ee118d95 Edgar E. Iglesias
    struct xlx_uartlite *s = FROM_SYSBUS(typeof (*s), dev);
198 ee118d95 Edgar E. Iglesias
199 ee118d95 Edgar E. Iglesias
    sysbus_init_irq(dev, &s->irq);
200 ee118d95 Edgar E. Iglesias
201 ee118d95 Edgar E. Iglesias
    uart_update_status(s);
202 23d6055e Peter A. G. Crosthwaite
    memory_region_init_io(&s->mmio, &uart_ops, s, "xlnx.xps-uartlite",
203 23d6055e Peter A. G. Crosthwaite
                                                                R_MAX * 4);
204 750ecd44 Avi Kivity
    sysbus_init_mmio(dev, &s->mmio);
205 ee118d95 Edgar E. Iglesias
206 0beb4942 Anthony Liguori
    s->chr = qemu_char_get_next_serial();
207 ee118d95 Edgar E. Iglesias
    if (s->chr)
208 ee118d95 Edgar E. Iglesias
        qemu_chr_add_handlers(s->chr, uart_can_rx, uart_rx, uart_event, s);
209 81a322d4 Gerd Hoffmann
    return 0;
210 ee118d95 Edgar E. Iglesias
}
211 ee118d95 Edgar E. Iglesias
212 999e12bb Anthony Liguori
static void xilinx_uartlite_class_init(ObjectClass *klass, void *data)
213 999e12bb Anthony Liguori
{
214 999e12bb Anthony Liguori
    SysBusDeviceClass *sdc = SYS_BUS_DEVICE_CLASS(klass);
215 999e12bb Anthony Liguori
216 999e12bb Anthony Liguori
    sdc->init = xilinx_uartlite_init;
217 999e12bb Anthony Liguori
}
218 999e12bb Anthony Liguori
219 8c43a6f0 Andreas Färber
static const TypeInfo xilinx_uartlite_info = {
220 23d6055e Peter A. G. Crosthwaite
    .name          = "xlnx.xps-uartlite",
221 39bffca2 Anthony Liguori
    .parent        = TYPE_SYS_BUS_DEVICE,
222 39bffca2 Anthony Liguori
    .instance_size = sizeof (struct xlx_uartlite),
223 39bffca2 Anthony Liguori
    .class_init    = xilinx_uartlite_class_init,
224 999e12bb Anthony Liguori
};
225 999e12bb Anthony Liguori
226 83f7d43a Andreas Färber
static void xilinx_uart_register_types(void)
227 ee118d95 Edgar E. Iglesias
{
228 39bffca2 Anthony Liguori
    type_register_static(&xilinx_uartlite_info);
229 ee118d95 Edgar E. Iglesias
}
230 ee118d95 Edgar E. Iglesias
231 83f7d43a Andreas Färber
type_init(xilinx_uart_register_types)