Commit 81174dae3f9189519cd60c7b79e91c291b021bbe
1 parent
06057e6f
Upgrade emulated UART to 16550A (Stefano Stabellini)
This patch upgrades the emulated UART to 16550A, the code comes from xen-unstable. The main improvement was introduced with the following patch and subsequent email thread: http://lists.xensource.com/archives/html/xen-devel/2007-12/msg00129.html The changes compared to previous version are: - change clock_gettime to qemu_get_clock - no token bucket anymore; - fixed a small bug handling IRQs; this was the problem that prevented kgdb to work over the serial (thanks to Jason Wessel for the help spotting and reproducing this bug). - many many style fixes; - savevm version number increased; - not including termios.h and sys/ioctl.h anymore, declaring static constants in qemu-char.h instead; Signed-off-by: Stefano Stabellini <stefano.stabellini@eu.citrix.com> Signed-off-by: Anthony Liguori <aliguori@us.ibm.com> git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@4993 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
3 changed files
with
433 additions
and
94 deletions
hw/serial.c
| 1 | /* | 1 | /* |
| 2 | - * QEMU 16450 UART emulation | 2 | + * QEMU 16550A UART emulation |
| 3 | * | 3 | * |
| 4 | * Copyright (c) 2003-2004 Fabrice Bellard | 4 | * Copyright (c) 2003-2004 Fabrice Bellard |
| 5 | + * Copyright (c) 2008 Citrix Systems, Inc. | ||
| 5 | * | 6 | * |
| 6 | * Permission is hereby granted, free of charge, to any person obtaining a copy | 7 | * Permission is hereby granted, free of charge, to any person obtaining a copy |
| 7 | * of this software and associated documentation files (the "Software"), to deal | 8 | * of this software and associated documentation files (the "Software"), to deal |
| @@ -43,6 +44,10 @@ | @@ -43,6 +44,10 @@ | ||
| 43 | #define UART_IIR_THRI 0x02 /* Transmitter holding register empty */ | 44 | #define UART_IIR_THRI 0x02 /* Transmitter holding register empty */ |
| 44 | #define UART_IIR_RDI 0x04 /* Receiver data interrupt */ | 45 | #define UART_IIR_RDI 0x04 /* Receiver data interrupt */ |
| 45 | #define UART_IIR_RLSI 0x06 /* Receiver line status interrupt */ | 46 | #define UART_IIR_RLSI 0x06 /* Receiver line status interrupt */ |
| 47 | +#define UART_IIR_CTI 0x0C /* Character Timeout Indication */ | ||
| 48 | + | ||
| 49 | +#define UART_IIR_FENF 0x80 /* Fifo enabled, but not functionning */ | ||
| 50 | +#define UART_IIR_FE 0xC0 /* Fifo enabled */ | ||
| 46 | 51 | ||
| 47 | /* | 52 | /* |
| 48 | * These are the definitions for the Modem Control Register | 53 | * These are the definitions for the Modem Control Register |
| @@ -73,17 +78,39 @@ | @@ -73,17 +78,39 @@ | ||
| 73 | #define UART_LSR_PE 0x04 /* Parity error indicator */ | 78 | #define UART_LSR_PE 0x04 /* Parity error indicator */ |
| 74 | #define UART_LSR_OE 0x02 /* Overrun error indicator */ | 79 | #define UART_LSR_OE 0x02 /* Overrun error indicator */ |
| 75 | #define UART_LSR_DR 0x01 /* Receiver data ready */ | 80 | #define UART_LSR_DR 0x01 /* Receiver data ready */ |
| 81 | +#define UART_LSR_INT_ANY 0x1E /* Any of the lsr-interrupt-triggering status bits */ | ||
| 76 | 82 | ||
| 77 | -/* | ||
| 78 | - * Delay TX IRQ after sending as much characters as the given interval would | ||
| 79 | - * contain on real hardware. This avoids overloading the guest if it processes | ||
| 80 | - * its output buffer in a loop inside the TX IRQ handler. | ||
| 81 | - */ | ||
| 82 | -#define THROTTLE_TX_INTERVAL 10 /* ms */ | 83 | +/* Interrupt trigger levels. The byte-counts are for 16550A - in newer UARTs the byte-count for each ITL is higher. */ |
| 84 | + | ||
| 85 | +#define UART_FCR_ITL_1 0x00 /* 1 byte ITL */ | ||
| 86 | +#define UART_FCR_ITL_2 0x40 /* 4 bytes ITL */ | ||
| 87 | +#define UART_FCR_ITL_3 0x80 /* 8 bytes ITL */ | ||
| 88 | +#define UART_FCR_ITL_4 0xC0 /* 14 bytes ITL */ | ||
| 89 | + | ||
| 90 | +#define UART_FCR_DMS 0x08 /* DMA Mode Select */ | ||
| 91 | +#define UART_FCR_XFR 0x04 /* XMIT Fifo Reset */ | ||
| 92 | +#define UART_FCR_RFR 0x02 /* RCVR Fifo Reset */ | ||
| 93 | +#define UART_FCR_FE 0x01 /* FIFO Enable */ | ||
| 94 | + | ||
| 95 | +#define UART_FIFO_LENGTH 16 /* 16550A Fifo Length */ | ||
| 96 | + | ||
| 97 | +#define XMIT_FIFO 0 | ||
| 98 | +#define RECV_FIFO 1 | ||
| 99 | +#define MAX_XMIT_RETRY 4 | ||
| 100 | + | ||
| 101 | +struct SerialFIFO { | ||
| 102 | + uint8_t data[UART_FIFO_LENGTH]; | ||
| 103 | + uint8_t count; | ||
| 104 | + uint8_t itl; /* Interrupt Trigger Level */ | ||
| 105 | + uint8_t tail; | ||
| 106 | + uint8_t head; | ||
| 107 | +} typedef SerialFIFO; | ||
| 83 | 108 | ||
| 84 | struct SerialState { | 109 | struct SerialState { |
| 85 | uint16_t divider; | 110 | uint16_t divider; |
| 86 | uint8_t rbr; /* receive register */ | 111 | uint8_t rbr; /* receive register */ |
| 112 | + uint8_t thr; /* transmit holding register */ | ||
| 113 | + uint8_t tsr; /* transmit shift register */ | ||
| 87 | uint8_t ier; | 114 | uint8_t ier; |
| 88 | uint8_t iir; /* read only */ | 115 | uint8_t iir; /* read only */ |
| 89 | uint8_t lcr; | 116 | uint8_t lcr; |
| @@ -91,6 +118,7 @@ struct SerialState { | @@ -91,6 +118,7 @@ struct SerialState { | ||
| 91 | uint8_t lsr; /* read only */ | 118 | uint8_t lsr; /* read only */ |
| 92 | uint8_t msr; /* read only */ | 119 | uint8_t msr; /* read only */ |
| 93 | uint8_t scr; | 120 | uint8_t scr; |
| 121 | + uint8_t fcr; | ||
| 94 | /* NOTE: this hidden state is necessary for tx irq generation as | 122 | /* NOTE: this hidden state is necessary for tx irq generation as |
| 95 | it can be reset while reading iir */ | 123 | it can be reset while reading iir */ |
| 96 | int thr_ipending; | 124 | int thr_ipending; |
| @@ -100,55 +128,106 @@ struct SerialState { | @@ -100,55 +128,106 @@ struct SerialState { | ||
| 100 | target_phys_addr_t base; | 128 | target_phys_addr_t base; |
| 101 | int it_shift; | 129 | int it_shift; |
| 102 | int baudbase; | 130 | int baudbase; |
| 103 | - QEMUTimer *tx_timer; | ||
| 104 | - int tx_burst; | 131 | + int tsr_retry; |
| 132 | + | ||
| 133 | + uint64_t last_xmit_ts; /* Time when the last byte was successfully sent out of the tsr */ | ||
| 134 | + SerialFIFO recv_fifo; | ||
| 135 | + SerialFIFO xmit_fifo; | ||
| 136 | + | ||
| 137 | + struct QEMUTimer *fifo_timeout_timer; | ||
| 138 | + int timeout_ipending; /* timeout interrupt pending state */ | ||
| 139 | + struct QEMUTimer *transmit_timer; | ||
| 140 | + | ||
| 141 | + | ||
| 142 | + uint64_t char_transmit_time; /* time to transmit a char in ticks*/ | ||
| 143 | + int poll_msl; | ||
| 144 | + | ||
| 145 | + struct QEMUTimer *modem_status_poll; | ||
| 105 | }; | 146 | }; |
| 106 | 147 | ||
| 107 | -static void serial_receive_byte(SerialState *s, int ch); | 148 | +static void serial_receive1(void *opaque, const uint8_t *buf, int size); |
| 108 | 149 | ||
| 109 | -static void serial_update_irq(SerialState *s) | 150 | +static void fifo_clear(SerialState *s, int fifo) |
| 110 | { | 151 | { |
| 111 | - if ((s->lsr & UART_LSR_DR) && (s->ier & UART_IER_RDI)) { | ||
| 112 | - s->iir = UART_IIR_RDI; | ||
| 113 | - } else if (s->thr_ipending && (s->ier & UART_IER_THRI)) { | ||
| 114 | - s->iir = UART_IIR_THRI; | ||
| 115 | - } else { | ||
| 116 | - s->iir = UART_IIR_NO_INT; | ||
| 117 | - } | ||
| 118 | - if (s->iir != UART_IIR_NO_INT) { | ||
| 119 | - qemu_irq_raise(s->irq); | ||
| 120 | - } else { | ||
| 121 | - qemu_irq_lower(s->irq); | ||
| 122 | - } | 152 | + SerialFIFO *f = (fifo) ? &s->recv_fifo : &s->xmit_fifo; |
| 153 | + memset(f->data, 0, UART_FIFO_LENGTH); | ||
| 154 | + f->count = 0; | ||
| 155 | + f->head = 0; | ||
| 156 | + f->tail = 0; | ||
| 123 | } | 157 | } |
| 124 | 158 | ||
| 125 | -static void serial_tx_done(void *opaque) | 159 | +static int fifo_put(SerialState *s, int fifo, uint8_t chr) |
| 126 | { | 160 | { |
| 127 | - SerialState *s = opaque; | 161 | + SerialFIFO *f = (fifo) ? &s->recv_fifo : &s->xmit_fifo; |
| 128 | 162 | ||
| 129 | - if (s->tx_burst < 0) { | ||
| 130 | - uint16_t divider; | 163 | + f->data[f->head++] = chr; |
| 131 | 164 | ||
| 132 | - if (s->divider) | ||
| 133 | - divider = s->divider; | ||
| 134 | - else | ||
| 135 | - divider = 1; | 165 | + if (f->head == UART_FIFO_LENGTH) |
| 166 | + f->head = 0; | ||
| 167 | + f->count++; | ||
| 168 | + | ||
| 169 | + return 1; | ||
| 170 | +} | ||
| 171 | + | ||
| 172 | +static uint8_t fifo_get(SerialState *s, int fifo) | ||
| 173 | +{ | ||
| 174 | + SerialFIFO *f = (fifo) ? &s->recv_fifo : &s->xmit_fifo; | ||
| 175 | + uint8_t c; | ||
| 176 | + | ||
| 177 | + if(f->count == 0) | ||
| 178 | + return 0; | ||
| 179 | + | ||
| 180 | + c = f->data[f->tail++]; | ||
| 181 | + if (f->tail == UART_FIFO_LENGTH) | ||
| 182 | + f->tail = 0; | ||
| 183 | + f->count--; | ||
| 184 | + | ||
| 185 | + return c; | ||
| 186 | +} | ||
| 136 | 187 | ||
| 137 | - /* We assume 10 bits/char, OK for this purpose. */ | ||
| 138 | - s->tx_burst = THROTTLE_TX_INTERVAL * 1000 / | ||
| 139 | - (1000000 * 10 / (s->baudbase / divider)); | 188 | +static void serial_update_irq(SerialState *s) |
| 189 | +{ | ||
| 190 | + uint8_t tmp_iir = UART_IIR_NO_INT; | ||
| 191 | + | ||
| 192 | + if (!s->ier) { | ||
| 193 | + qemu_irq_lower(s->irq); | ||
| 194 | + return; | ||
| 195 | + } | ||
| 196 | + | ||
| 197 | + if ((s->ier & UART_IER_RLSI) && (s->lsr & UART_LSR_INT_ANY)) { | ||
| 198 | + tmp_iir = UART_IIR_RLSI; | ||
| 199 | + } else if (s->timeout_ipending) { | ||
| 200 | + tmp_iir = UART_IIR_CTI; | ||
| 201 | + } else if ((s->ier & UART_IER_RDI) && (s->lsr & UART_LSR_DR)) { | ||
| 202 | + if (!(s->fcr & UART_FCR_FE)) { | ||
| 203 | + tmp_iir = UART_IIR_RDI; | ||
| 204 | + } else if (s->recv_fifo.count >= s->recv_fifo.itl) { | ||
| 205 | + tmp_iir = UART_IIR_RDI; | ||
| 206 | + } | ||
| 207 | + } else if ((s->ier & UART_IER_THRI) && s->thr_ipending) { | ||
| 208 | + tmp_iir = UART_IIR_THRI; | ||
| 209 | + } else if ((s->ier & UART_IER_MSI) && (s->msr & UART_MSR_ANY_DELTA)) { | ||
| 210 | + tmp_iir = UART_IIR_MSI; | ||
| 211 | + } | ||
| 212 | + | ||
| 213 | + s->iir = tmp_iir | (s->iir & 0xF0); | ||
| 214 | + | ||
| 215 | + if (tmp_iir != UART_IIR_NO_INT) { | ||
| 216 | + qemu_irq_raise(s->irq); | ||
| 217 | + } else { | ||
| 218 | + qemu_irq_lower(s->irq); | ||
| 140 | } | 219 | } |
| 141 | - s->thr_ipending = 1; | ||
| 142 | - s->lsr |= UART_LSR_THRE; | ||
| 143 | - s->lsr |= UART_LSR_TEMT; | ||
| 144 | - serial_update_irq(s); | ||
| 145 | } | 220 | } |
| 146 | 221 | ||
| 147 | static void serial_update_parameters(SerialState *s) | 222 | static void serial_update_parameters(SerialState *s) |
| 148 | { | 223 | { |
| 149 | - int speed, parity, data_bits, stop_bits; | 224 | + int speed, parity, data_bits, stop_bits, frame_size; |
| 150 | QEMUSerialSetParams ssp; | 225 | QEMUSerialSetParams ssp; |
| 151 | 226 | ||
| 227 | + if (s->divider == 0) | ||
| 228 | + return; | ||
| 229 | + | ||
| 230 | + frame_size = 1; | ||
| 152 | if (s->lcr & 0x08) { | 231 | if (s->lcr & 0x08) { |
| 153 | if (s->lcr & 0x10) | 232 | if (s->lcr & 0x10) |
| 154 | parity = 'E'; | 233 | parity = 'E'; |
| @@ -156,19 +235,21 @@ static void serial_update_parameters(SerialState *s) | @@ -156,19 +235,21 @@ static void serial_update_parameters(SerialState *s) | ||
| 156 | parity = 'O'; | 235 | parity = 'O'; |
| 157 | } else { | 236 | } else { |
| 158 | parity = 'N'; | 237 | parity = 'N'; |
| 238 | + frame_size = 0; | ||
| 159 | } | 239 | } |
| 160 | if (s->lcr & 0x04) | 240 | if (s->lcr & 0x04) |
| 161 | stop_bits = 2; | 241 | stop_bits = 2; |
| 162 | else | 242 | else |
| 163 | stop_bits = 1; | 243 | stop_bits = 1; |
| 244 | + | ||
| 164 | data_bits = (s->lcr & 0x03) + 5; | 245 | data_bits = (s->lcr & 0x03) + 5; |
| 165 | - if (s->divider == 0) | ||
| 166 | - return; | 246 | + frame_size += data_bits + stop_bits; |
| 167 | speed = s->baudbase / s->divider; | 247 | speed = s->baudbase / s->divider; |
| 168 | ssp.speed = speed; | 248 | ssp.speed = speed; |
| 169 | ssp.parity = parity; | 249 | ssp.parity = parity; |
| 170 | ssp.data_bits = data_bits; | 250 | ssp.data_bits = data_bits; |
| 171 | ssp.stop_bits = stop_bits; | 251 | ssp.stop_bits = stop_bits; |
| 252 | + s->char_transmit_time = (ticks_per_sec / speed) * frame_size; | ||
| 172 | qemu_chr_ioctl(s->chr, CHR_IOCTL_SERIAL_SET_PARAMS, &ssp); | 253 | qemu_chr_ioctl(s->chr, CHR_IOCTL_SERIAL_SET_PARAMS, &ssp); |
| 173 | #if 0 | 254 | #if 0 |
| 174 | printf("speed=%d parity=%c data=%d stop=%d\n", | 255 | printf("speed=%d parity=%c data=%d stop=%d\n", |
| @@ -176,10 +257,91 @@ static void serial_update_parameters(SerialState *s) | @@ -176,10 +257,91 @@ static void serial_update_parameters(SerialState *s) | ||
| 176 | #endif | 257 | #endif |
| 177 | } | 258 | } |
| 178 | 259 | ||
| 260 | +static void serial_update_msl(SerialState *s) | ||
| 261 | +{ | ||
| 262 | + uint8_t omsr; | ||
| 263 | + int flags; | ||
| 264 | + | ||
| 265 | + qemu_del_timer(s->modem_status_poll); | ||
| 266 | + | ||
| 267 | + if (qemu_chr_ioctl(s->chr,CHR_IOCTL_SERIAL_GET_TIOCM, &flags) == -ENOTSUP) { | ||
| 268 | + s->poll_msl = -1; | ||
| 269 | + return; | ||
| 270 | + } | ||
| 271 | + | ||
| 272 | + omsr = s->msr; | ||
| 273 | + | ||
| 274 | + s->msr = (flags & CHR_TIOCM_CTS) ? s->msr | UART_MSR_CTS : s->msr & ~UART_MSR_CTS; | ||
| 275 | + s->msr = (flags & CHR_TIOCM_DSR) ? s->msr | UART_MSR_DSR : s->msr & ~UART_MSR_DSR; | ||
| 276 | + s->msr = (flags & CHR_TIOCM_CAR) ? s->msr | UART_MSR_DCD : s->msr & ~UART_MSR_DCD; | ||
| 277 | + s->msr = (flags & CHR_TIOCM_RI) ? s->msr | UART_MSR_RI : s->msr & ~UART_MSR_RI; | ||
| 278 | + | ||
| 279 | + if (s->msr != omsr) { | ||
| 280 | + /* Set delta bits */ | ||
| 281 | + s->msr = s->msr | ((s->msr >> 4) ^ (omsr >> 4)); | ||
| 282 | + /* UART_MSR_TERI only if change was from 1 -> 0 */ | ||
| 283 | + if ((s->msr & UART_MSR_TERI) && !(omsr & UART_MSR_RI)) | ||
| 284 | + s->msr &= ~UART_MSR_TERI; | ||
| 285 | + serial_update_irq(s); | ||
| 286 | + } | ||
| 287 | + | ||
| 288 | + /* The real 16550A apparently has a 250ns response latency to line status changes. | ||
| 289 | + We'll be lazy and poll only every 10ms, and only poll it at all if MSI interrupts are turned on */ | ||
| 290 | + | ||
| 291 | + if (s->poll_msl) | ||
| 292 | + qemu_mod_timer(s->modem_status_poll, qemu_get_clock(vm_clock) + ticks_per_sec / 100); | ||
| 293 | +} | ||
| 294 | + | ||
| 295 | +static void serial_xmit(void *opaque) | ||
| 296 | +{ | ||
| 297 | + SerialState *s = opaque; | ||
| 298 | + uint64_t new_xmit_ts = qemu_get_clock(vm_clock); | ||
| 299 | + | ||
| 300 | + if (s->tsr_retry <= 0) { | ||
| 301 | + if (s->fcr & UART_FCR_FE) { | ||
| 302 | + s->tsr = fifo_get(s,XMIT_FIFO); | ||
| 303 | + if (!s->xmit_fifo.count) | ||
| 304 | + s->lsr |= UART_LSR_THRE; | ||
| 305 | + } else { | ||
| 306 | + s->tsr = s->thr; | ||
| 307 | + s->lsr |= UART_LSR_THRE; | ||
| 308 | + } | ||
| 309 | + } | ||
| 310 | + | ||
| 311 | + if (s->mcr & UART_MCR_LOOP) { | ||
| 312 | + /* in loopback mode, say that we just received a char */ | ||
| 313 | + serial_receive1(s, &s->tsr, 1); | ||
| 314 | + } else if (qemu_chr_write(s->chr, &s->tsr, 1) != 1) { | ||
| 315 | + if ((s->tsr_retry > 0) && (s->tsr_retry <= MAX_XMIT_RETRY)) { | ||
| 316 | + s->tsr_retry++; | ||
| 317 | + qemu_mod_timer(s->transmit_timer, new_xmit_ts + s->char_transmit_time); | ||
| 318 | + return; | ||
| 319 | + } else if (s->poll_msl < 0) { | ||
| 320 | + /* If we exceed MAX_XMIT_RETRY and the backend is not a real serial port, then | ||
| 321 | + drop any further failed writes instantly, until we get one that goes through. | ||
| 322 | + This is to prevent guests that log to unconnected pipes or pty's from stalling. */ | ||
| 323 | + s->tsr_retry = -1; | ||
| 324 | + } | ||
| 325 | + } | ||
| 326 | + else { | ||
| 327 | + s->tsr_retry = 0; | ||
| 328 | + } | ||
| 329 | + | ||
| 330 | + s->last_xmit_ts = qemu_get_clock(vm_clock); | ||
| 331 | + if (!(s->lsr & UART_LSR_THRE)) | ||
| 332 | + qemu_mod_timer(s->transmit_timer, s->last_xmit_ts + s->char_transmit_time); | ||
| 333 | + | ||
| 334 | + if (s->lsr & UART_LSR_THRE) { | ||
| 335 | + s->lsr |= UART_LSR_TEMT; | ||
| 336 | + s->thr_ipending = 1; | ||
| 337 | + serial_update_irq(s); | ||
| 338 | + } | ||
| 339 | +} | ||
| 340 | + | ||
| 341 | + | ||
| 179 | static void serial_ioport_write(void *opaque, uint32_t addr, uint32_t val) | 342 | static void serial_ioport_write(void *opaque, uint32_t addr, uint32_t val) |
| 180 | { | 343 | { |
| 181 | SerialState *s = opaque; | 344 | SerialState *s = opaque; |
| 182 | - unsigned char ch; | ||
| 183 | 345 | ||
| 184 | addr &= 7; | 346 | addr &= 7; |
| 185 | #ifdef DEBUG_SERIAL | 347 | #ifdef DEBUG_SERIAL |
| @@ -192,25 +354,19 @@ static void serial_ioport_write(void *opaque, uint32_t addr, uint32_t val) | @@ -192,25 +354,19 @@ static void serial_ioport_write(void *opaque, uint32_t addr, uint32_t val) | ||
| 192 | s->divider = (s->divider & 0xff00) | val; | 354 | s->divider = (s->divider & 0xff00) | val; |
| 193 | serial_update_parameters(s); | 355 | serial_update_parameters(s); |
| 194 | } else { | 356 | } else { |
| 357 | + s->thr = (uint8_t) val; | ||
| 358 | + if(s->fcr & UART_FCR_FE) { | ||
| 359 | + fifo_put(s, XMIT_FIFO, s->thr); | ||
| 195 | s->thr_ipending = 0; | 360 | s->thr_ipending = 0; |
| 361 | + s->lsr &= ~UART_LSR_TEMT; | ||
| 196 | s->lsr &= ~UART_LSR_THRE; | 362 | s->lsr &= ~UART_LSR_THRE; |
| 197 | serial_update_irq(s); | 363 | serial_update_irq(s); |
| 198 | - ch = val; | ||
| 199 | - if (!(s->mcr & UART_MCR_LOOP)) { | ||
| 200 | - /* when not in loopback mode, send the char */ | ||
| 201 | - qemu_chr_write(s->chr, &ch, 1); | ||
| 202 | } else { | 364 | } else { |
| 203 | - /* in loopback mode, say that we just received a char */ | ||
| 204 | - serial_receive_byte(s, ch); | ||
| 205 | - } | ||
| 206 | - if (s->tx_burst > 0) { | ||
| 207 | - s->tx_burst--; | ||
| 208 | - serial_tx_done(s); | ||
| 209 | - } else if (s->tx_burst == 0) { | ||
| 210 | - s->tx_burst--; | ||
| 211 | - qemu_mod_timer(s->tx_timer, qemu_get_clock(vm_clock) + | ||
| 212 | - ticks_per_sec * THROTTLE_TX_INTERVAL / 1000); | 365 | + s->thr_ipending = 0; |
| 366 | + s->lsr &= ~UART_LSR_THRE; | ||
| 367 | + serial_update_irq(s); | ||
| 213 | } | 368 | } |
| 369 | + serial_xmit(s); | ||
| 214 | } | 370 | } |
| 215 | break; | 371 | break; |
| 216 | case 1: | 372 | case 1: |
| @@ -219,13 +375,68 @@ static void serial_ioport_write(void *opaque, uint32_t addr, uint32_t val) | @@ -219,13 +375,68 @@ static void serial_ioport_write(void *opaque, uint32_t addr, uint32_t val) | ||
| 219 | serial_update_parameters(s); | 375 | serial_update_parameters(s); |
| 220 | } else { | 376 | } else { |
| 221 | s->ier = val & 0x0f; | 377 | s->ier = val & 0x0f; |
| 378 | + /* If the backend device is a real serial port, turn polling of the modem | ||
| 379 | + status lines on physical port on or off depending on UART_IER_MSI state */ | ||
| 380 | + if (s->poll_msl >= 0) { | ||
| 381 | + if (s->ier & UART_IER_MSI) { | ||
| 382 | + s->poll_msl = 1; | ||
| 383 | + serial_update_msl(s); | ||
| 384 | + } else { | ||
| 385 | + qemu_del_timer(s->modem_status_poll); | ||
| 386 | + s->poll_msl = 0; | ||
| 387 | + } | ||
| 388 | + } | ||
| 222 | if (s->lsr & UART_LSR_THRE) { | 389 | if (s->lsr & UART_LSR_THRE) { |
| 223 | s->thr_ipending = 1; | 390 | s->thr_ipending = 1; |
| 391 | + serial_update_irq(s); | ||
| 224 | } | 392 | } |
| 225 | - serial_update_irq(s); | ||
| 226 | } | 393 | } |
| 227 | break; | 394 | break; |
| 228 | case 2: | 395 | case 2: |
| 396 | + val = val & 0xFF; | ||
| 397 | + | ||
| 398 | + if (s->fcr == val) | ||
| 399 | + break; | ||
| 400 | + | ||
| 401 | + /* Did the enable/disable flag change? If so, make sure FIFOs get flushed */ | ||
| 402 | + if ((val ^ s->fcr) & UART_FCR_FE) | ||
| 403 | + val |= UART_FCR_XFR | UART_FCR_RFR; | ||
| 404 | + | ||
| 405 | + /* FIFO clear */ | ||
| 406 | + | ||
| 407 | + if (val & UART_FCR_RFR) { | ||
| 408 | + qemu_del_timer(s->fifo_timeout_timer); | ||
| 409 | + s->timeout_ipending=0; | ||
| 410 | + fifo_clear(s,RECV_FIFO); | ||
| 411 | + } | ||
| 412 | + | ||
| 413 | + if (val & UART_FCR_XFR) { | ||
| 414 | + fifo_clear(s,XMIT_FIFO); | ||
| 415 | + } | ||
| 416 | + | ||
| 417 | + if (val & UART_FCR_FE) { | ||
| 418 | + s->iir |= UART_IIR_FE; | ||
| 419 | + /* Set RECV_FIFO trigger Level */ | ||
| 420 | + switch (val & 0xC0) { | ||
| 421 | + case UART_FCR_ITL_1: | ||
| 422 | + s->recv_fifo.itl = 1; | ||
| 423 | + break; | ||
| 424 | + case UART_FCR_ITL_2: | ||
| 425 | + s->recv_fifo.itl = 4; | ||
| 426 | + break; | ||
| 427 | + case UART_FCR_ITL_3: | ||
| 428 | + s->recv_fifo.itl = 8; | ||
| 429 | + break; | ||
| 430 | + case UART_FCR_ITL_4: | ||
| 431 | + s->recv_fifo.itl = 14; | ||
| 432 | + break; | ||
| 433 | + } | ||
| 434 | + } else | ||
| 435 | + s->iir &= ~UART_IIR_FE; | ||
| 436 | + | ||
| 437 | + /* Set fcr - or at least the bits in it that are supposed to "stick" */ | ||
| 438 | + s->fcr = val & 0xC9; | ||
| 439 | + serial_update_irq(s); | ||
| 229 | break; | 440 | break; |
| 230 | case 3: | 441 | case 3: |
| 231 | { | 442 | { |
| @@ -241,7 +452,30 @@ static void serial_ioport_write(void *opaque, uint32_t addr, uint32_t val) | @@ -241,7 +452,30 @@ static void serial_ioport_write(void *opaque, uint32_t addr, uint32_t val) | ||
| 241 | } | 452 | } |
| 242 | break; | 453 | break; |
| 243 | case 4: | 454 | case 4: |
| 244 | - s->mcr = val & 0x1f; | 455 | + { |
| 456 | + int flags; | ||
| 457 | + int old_mcr = s->mcr; | ||
| 458 | + s->mcr = val & 0x1f; | ||
| 459 | + if (val & UART_MCR_LOOP) | ||
| 460 | + break; | ||
| 461 | + | ||
| 462 | + if (s->poll_msl >= 0 && old_mcr != s->mcr) { | ||
| 463 | + | ||
| 464 | + qemu_chr_ioctl(s->chr,CHR_IOCTL_SERIAL_GET_TIOCM, &flags); | ||
| 465 | + | ||
| 466 | + flags &= ~(CHR_TIOCM_RTS | CHR_TIOCM_DTR); | ||
| 467 | + | ||
| 468 | + if (val & UART_MCR_RTS) | ||
| 469 | + flags |= CHR_TIOCM_RTS; | ||
| 470 | + if (val & UART_MCR_DTR) | ||
| 471 | + flags |= CHR_TIOCM_DTR; | ||
| 472 | + | ||
| 473 | + qemu_chr_ioctl(s->chr,CHR_IOCTL_SERIAL_SET_TIOCM, &flags); | ||
| 474 | + /* Update the modem status after a one-character-send wait-time, since there may be a response | ||
| 475 | + from the device/computer at the other end of the serial line */ | ||
| 476 | + qemu_mod_timer(s->modem_status_poll, qemu_get_clock(vm_clock) + s->char_transmit_time); | ||
| 477 | + } | ||
| 478 | + } | ||
| 245 | break; | 479 | break; |
| 246 | case 5: | 480 | case 5: |
| 247 | break; | 481 | break; |
| @@ -265,8 +499,17 @@ static uint32_t serial_ioport_read(void *opaque, uint32_t addr) | @@ -265,8 +499,17 @@ static uint32_t serial_ioport_read(void *opaque, uint32_t addr) | ||
| 265 | if (s->lcr & UART_LCR_DLAB) { | 499 | if (s->lcr & UART_LCR_DLAB) { |
| 266 | ret = s->divider & 0xff; | 500 | ret = s->divider & 0xff; |
| 267 | } else { | 501 | } else { |
| 268 | - ret = s->rbr; | ||
| 269 | - s->lsr &= ~(UART_LSR_DR | UART_LSR_BI); | 502 | + if(s->fcr & UART_FCR_FE) { |
| 503 | + ret = fifo_get(s,RECV_FIFO); | ||
| 504 | + if (s->recv_fifo.count == 0) | ||
| 505 | + s->lsr &= ~(UART_LSR_DR | UART_LSR_BI); | ||
| 506 | + else | ||
| 507 | + qemu_mod_timer(s->fifo_timeout_timer, qemu_get_clock (vm_clock) + s->char_transmit_time * 4); | ||
| 508 | + s->timeout_ipending = 0; | ||
| 509 | + } else { | ||
| 510 | + ret = s->rbr; | ||
| 511 | + s->lsr &= ~(UART_LSR_DR | UART_LSR_BI); | ||
| 512 | + } | ||
| 270 | serial_update_irq(s); | 513 | serial_update_irq(s); |
| 271 | if (!(s->mcr & UART_MCR_LOOP)) { | 514 | if (!(s->mcr & UART_MCR_LOOP)) { |
| 272 | /* in loopback mode, don't receive any data */ | 515 | /* in loopback mode, don't receive any data */ |
| @@ -283,8 +526,6 @@ static uint32_t serial_ioport_read(void *opaque, uint32_t addr) | @@ -283,8 +526,6 @@ static uint32_t serial_ioport_read(void *opaque, uint32_t addr) | ||
| 283 | break; | 526 | break; |
| 284 | case 2: | 527 | case 2: |
| 285 | ret = s->iir; | 528 | ret = s->iir; |
| 286 | - /* reset THR pending bit */ | ||
| 287 | - if ((ret & 0x7) == UART_IIR_THRI) | ||
| 288 | s->thr_ipending = 0; | 529 | s->thr_ipending = 0; |
| 289 | serial_update_irq(s); | 530 | serial_update_irq(s); |
| 290 | break; | 531 | break; |
| @@ -296,6 +537,11 @@ static uint32_t serial_ioport_read(void *opaque, uint32_t addr) | @@ -296,6 +537,11 @@ static uint32_t serial_ioport_read(void *opaque, uint32_t addr) | ||
| 296 | break; | 537 | break; |
| 297 | case 5: | 538 | case 5: |
| 298 | ret = s->lsr; | 539 | ret = s->lsr; |
| 540 | + /* Clear break interrupt */ | ||
| 541 | + if (s->lsr & UART_LSR_BI) { | ||
| 542 | + s->lsr &= ~UART_LSR_BI; | ||
| 543 | + serial_update_irq(s); | ||
| 544 | + } | ||
| 299 | break; | 545 | break; |
| 300 | case 6: | 546 | case 6: |
| 301 | if (s->mcr & UART_MCR_LOOP) { | 547 | if (s->mcr & UART_MCR_LOOP) { |
| @@ -305,7 +551,14 @@ static uint32_t serial_ioport_read(void *opaque, uint32_t addr) | @@ -305,7 +551,14 @@ static uint32_t serial_ioport_read(void *opaque, uint32_t addr) | ||
| 305 | ret |= (s->mcr & 0x02) << 3; | 551 | ret |= (s->mcr & 0x02) << 3; |
| 306 | ret |= (s->mcr & 0x01) << 5; | 552 | ret |= (s->mcr & 0x01) << 5; |
| 307 | } else { | 553 | } else { |
| 554 | + if (s->poll_msl >= 0) | ||
| 555 | + serial_update_msl(s); | ||
| 308 | ret = s->msr; | 556 | ret = s->msr; |
| 557 | + /* Clear delta bits & msr int after read, if they were set */ | ||
| 558 | + if (s->msr & UART_MSR_ANY_DELTA) { | ||
| 559 | + s->msr &= 0xF0; | ||
| 560 | + serial_update_irq(s); | ||
| 561 | + } | ||
| 309 | } | 562 | } |
| 310 | break; | 563 | break; |
| 311 | case 7: | 564 | case 7: |
| @@ -320,14 +573,17 @@ static uint32_t serial_ioport_read(void *opaque, uint32_t addr) | @@ -320,14 +573,17 @@ static uint32_t serial_ioport_read(void *opaque, uint32_t addr) | ||
| 320 | 573 | ||
| 321 | static int serial_can_receive(SerialState *s) | 574 | static int serial_can_receive(SerialState *s) |
| 322 | { | 575 | { |
| 576 | + if(s->fcr & UART_FCR_FE) { | ||
| 577 | + if(s->recv_fifo.count < UART_FIFO_LENGTH) | ||
| 578 | + /* Advertise (fifo.itl - fifo.count) bytes when count < ITL, and 1 if above. If UART_FIFO_LENGTH - fifo.count is | ||
| 579 | + advertised the effect will be to almost always fill the fifo completely before the guest has a chance to respond, | ||
| 580 | + effectively overriding the ITL that the guest has set. */ | ||
| 581 | + return (s->recv_fifo.count <= s->recv_fifo.itl) ? s->recv_fifo.itl - s->recv_fifo.count : 1; | ||
| 582 | + else | ||
| 583 | + return 0; | ||
| 584 | + } else { | ||
| 323 | return !(s->lsr & UART_LSR_DR); | 585 | return !(s->lsr & UART_LSR_DR); |
| 324 | -} | ||
| 325 | - | ||
| 326 | -static void serial_receive_byte(SerialState *s, int ch) | ||
| 327 | -{ | ||
| 328 | - s->rbr = ch; | ||
| 329 | - s->lsr |= UART_LSR_DR; | ||
| 330 | - serial_update_irq(s); | 586 | + } |
| 331 | } | 587 | } |
| 332 | 588 | ||
| 333 | static void serial_receive_break(SerialState *s) | 589 | static void serial_receive_break(SerialState *s) |
| @@ -337,6 +593,15 @@ static void serial_receive_break(SerialState *s) | @@ -337,6 +593,15 @@ static void serial_receive_break(SerialState *s) | ||
| 337 | serial_update_irq(s); | 593 | serial_update_irq(s); |
| 338 | } | 594 | } |
| 339 | 595 | ||
| 596 | +/* There's data in recv_fifo and s->rbr has not been read for 4 char transmit times */ | ||
| 597 | +static void fifo_timeout_int (void *opaque) { | ||
| 598 | + SerialState *s = opaque; | ||
| 599 | + if (s->recv_fifo.count) { | ||
| 600 | + s->timeout_ipending = 1; | ||
| 601 | + serial_update_irq(s); | ||
| 602 | + } | ||
| 603 | +} | ||
| 604 | + | ||
| 340 | static int serial_can_receive1(void *opaque) | 605 | static int serial_can_receive1(void *opaque) |
| 341 | { | 606 | { |
| 342 | SerialState *s = opaque; | 607 | SerialState *s = opaque; |
| @@ -346,12 +611,27 @@ static int serial_can_receive1(void *opaque) | @@ -346,12 +611,27 @@ static int serial_can_receive1(void *opaque) | ||
| 346 | static void serial_receive1(void *opaque, const uint8_t *buf, int size) | 611 | static void serial_receive1(void *opaque, const uint8_t *buf, int size) |
| 347 | { | 612 | { |
| 348 | SerialState *s = opaque; | 613 | SerialState *s = opaque; |
| 349 | - serial_receive_byte(s, buf[0]); | 614 | + if(s->fcr & UART_FCR_FE) { |
| 615 | + int i; | ||
| 616 | + for (i = 0; i < size; i++) { | ||
| 617 | + fifo_put(s, RECV_FIFO, buf[i]); | ||
| 618 | + } | ||
| 619 | + s->lsr |= UART_LSR_DR; | ||
| 620 | + /* call the timeout receive callback in 4 char transmit time */ | ||
| 621 | + qemu_mod_timer(s->fifo_timeout_timer, qemu_get_clock (vm_clock) + s->char_transmit_time * 4); | ||
| 622 | + } else { | ||
| 623 | + s->rbr = buf[0]; | ||
| 624 | + s->lsr |= UART_LSR_DR; | ||
| 625 | + } | ||
| 626 | + serial_update_irq(s); | ||
| 350 | } | 627 | } |
| 351 | 628 | ||
| 352 | static void serial_event(void *opaque, int event) | 629 | static void serial_event(void *opaque, int event) |
| 353 | { | 630 | { |
| 354 | SerialState *s = opaque; | 631 | SerialState *s = opaque; |
| 632 | +#ifdef DEBUG_SERIAL | ||
| 633 | + printf("serial: event %x\n", event); | ||
| 634 | +#endif | ||
| 355 | if (event == CHR_EVENT_BREAK) | 635 | if (event == CHR_EVENT_BREAK) |
| 356 | serial_receive_break(s); | 636 | serial_receive_break(s); |
| 357 | } | 637 | } |
| @@ -369,13 +649,15 @@ static void serial_save(QEMUFile *f, void *opaque) | @@ -369,13 +649,15 @@ static void serial_save(QEMUFile *f, void *opaque) | ||
| 369 | qemu_put_8s(f,&s->lsr); | 649 | qemu_put_8s(f,&s->lsr); |
| 370 | qemu_put_8s(f,&s->msr); | 650 | qemu_put_8s(f,&s->msr); |
| 371 | qemu_put_8s(f,&s->scr); | 651 | qemu_put_8s(f,&s->scr); |
| 652 | + qemu_put_8s(f,&s->fcr); | ||
| 372 | } | 653 | } |
| 373 | 654 | ||
| 374 | static int serial_load(QEMUFile *f, void *opaque, int version_id) | 655 | static int serial_load(QEMUFile *f, void *opaque, int version_id) |
| 375 | { | 656 | { |
| 376 | SerialState *s = opaque; | 657 | SerialState *s = opaque; |
| 658 | + uint8_t fcr = 0; | ||
| 377 | 659 | ||
| 378 | - if(version_id > 2) | 660 | + if(version_id > 3) |
| 379 | return -EINVAL; | 661 | return -EINVAL; |
| 380 | 662 | ||
| 381 | if (version_id >= 2) | 663 | if (version_id >= 2) |
| @@ -391,6 +673,11 @@ static int serial_load(QEMUFile *f, void *opaque, int version_id) | @@ -391,6 +673,11 @@ static int serial_load(QEMUFile *f, void *opaque, int version_id) | ||
| 391 | qemu_get_8s(f,&s->msr); | 673 | qemu_get_8s(f,&s->msr); |
| 392 | qemu_get_8s(f,&s->scr); | 674 | qemu_get_8s(f,&s->scr); |
| 393 | 675 | ||
| 676 | + if (version_id >= 3) | ||
| 677 | + qemu_get_8s(f,&fcr); | ||
| 678 | + | ||
| 679 | + /* Initialize fcr via setter to perform essential side-effects */ | ||
| 680 | + serial_ioport_write(s, 0x02, fcr); | ||
| 394 | return 0; | 681 | return 0; |
| 395 | } | 682 | } |
| 396 | 683 | ||
| @@ -398,21 +685,47 @@ static void serial_reset(void *opaque) | @@ -398,21 +685,47 @@ static void serial_reset(void *opaque) | ||
| 398 | { | 685 | { |
| 399 | SerialState *s = opaque; | 686 | SerialState *s = opaque; |
| 400 | 687 | ||
| 401 | - s->divider = 0; | ||
| 402 | s->rbr = 0; | 688 | s->rbr = 0; |
| 403 | s->ier = 0; | 689 | s->ier = 0; |
| 404 | s->iir = UART_IIR_NO_INT; | 690 | s->iir = UART_IIR_NO_INT; |
| 405 | s->lcr = 0; | 691 | s->lcr = 0; |
| 406 | - s->mcr = 0; | ||
| 407 | s->lsr = UART_LSR_TEMT | UART_LSR_THRE; | 692 | s->lsr = UART_LSR_TEMT | UART_LSR_THRE; |
| 408 | s->msr = UART_MSR_DCD | UART_MSR_DSR | UART_MSR_CTS; | 693 | s->msr = UART_MSR_DCD | UART_MSR_DSR | UART_MSR_CTS; |
| 694 | + /* Default to 9600 baud, no parity, one stop bit */ | ||
| 695 | + s->divider = 0x0C; | ||
| 696 | + s->mcr = UART_MCR_OUT2; | ||
| 409 | s->scr = 0; | 697 | s->scr = 0; |
| 698 | + s->tsr_retry = 0; | ||
| 699 | + s->char_transmit_time = (ticks_per_sec / 9600) * 9; | ||
| 700 | + s->poll_msl = 0; | ||
| 701 | + | ||
| 702 | + fifo_clear(s,RECV_FIFO); | ||
| 703 | + fifo_clear(s,XMIT_FIFO); | ||
| 704 | + | ||
| 705 | + s->last_xmit_ts = qemu_get_clock(vm_clock); | ||
| 410 | 706 | ||
| 411 | s->thr_ipending = 0; | 707 | s->thr_ipending = 0; |
| 412 | s->last_break_enable = 0; | 708 | s->last_break_enable = 0; |
| 413 | qemu_irq_lower(s->irq); | 709 | qemu_irq_lower(s->irq); |
| 414 | } | 710 | } |
| 415 | 711 | ||
| 712 | +static void serial_init_core(SerialState *s, qemu_irq irq, int baudbase, | ||
| 713 | + CharDriverState *chr) | ||
| 714 | +{ | ||
| 715 | + s->irq = irq; | ||
| 716 | + s->baudbase = baudbase; | ||
| 717 | + s->chr = chr; | ||
| 718 | + | ||
| 719 | + s->modem_status_poll = qemu_new_timer(vm_clock, (QEMUTimerCB *) serial_update_msl, s); | ||
| 720 | + | ||
| 721 | + s->fifo_timeout_timer = qemu_new_timer(vm_clock, (QEMUTimerCB *) fifo_timeout_int, s); | ||
| 722 | + s->transmit_timer = qemu_new_timer(vm_clock, (QEMUTimerCB *) serial_xmit, s); | ||
| 723 | + | ||
| 724 | + qemu_register_reset(serial_reset, s); | ||
| 725 | + serial_reset(s); | ||
| 726 | + | ||
| 727 | +} | ||
| 728 | + | ||
| 416 | /* If fd is zero, it means that the serial device uses the console */ | 729 | /* If fd is zero, it means that the serial device uses the console */ |
| 417 | SerialState *serial_init(int base, qemu_irq irq, int baudbase, | 730 | SerialState *serial_init(int base, qemu_irq irq, int baudbase, |
| 418 | CharDriverState *chr) | 731 | CharDriverState *chr) |
| @@ -422,21 +735,13 @@ SerialState *serial_init(int base, qemu_irq irq, int baudbase, | @@ -422,21 +735,13 @@ SerialState *serial_init(int base, qemu_irq irq, int baudbase, | ||
| 422 | s = qemu_mallocz(sizeof(SerialState)); | 735 | s = qemu_mallocz(sizeof(SerialState)); |
| 423 | if (!s) | 736 | if (!s) |
| 424 | return NULL; | 737 | return NULL; |
| 425 | - s->irq = irq; | ||
| 426 | - s->baudbase = baudbase; | ||
| 427 | - | ||
| 428 | - s->tx_timer = qemu_new_timer(vm_clock, serial_tx_done, s); | ||
| 429 | - if (!s->tx_timer) | ||
| 430 | - return NULL; | ||
| 431 | 738 | ||
| 432 | - qemu_register_reset(serial_reset, s); | ||
| 433 | - serial_reset(s); | 739 | + serial_init_core(s, irq, baudbase, chr); |
| 434 | 740 | ||
| 435 | - register_savevm("serial", base, 2, serial_save, serial_load, s); | 741 | + register_savevm("serial", base, 3, serial_save, serial_load, s); |
| 436 | 742 | ||
| 437 | register_ioport_write(base, 8, 1, serial_ioport_write, s); | 743 | register_ioport_write(base, 8, 1, serial_ioport_write, s); |
| 438 | register_ioport_read(base, 8, 1, serial_ioport_read, s); | 744 | register_ioport_read(base, 8, 1, serial_ioport_read, s); |
| 439 | - s->chr = chr; | ||
| 440 | qemu_chr_add_handlers(chr, serial_can_receive1, serial_receive1, | 745 | qemu_chr_add_handlers(chr, serial_can_receive1, serial_receive1, |
| 441 | serial_event, s); | 746 | serial_event, s); |
| 442 | return s; | 747 | return s; |
| @@ -524,27 +829,20 @@ SerialState *serial_mm_init (target_phys_addr_t base, int it_shift, | @@ -524,27 +829,20 @@ SerialState *serial_mm_init (target_phys_addr_t base, int it_shift, | ||
| 524 | s = qemu_mallocz(sizeof(SerialState)); | 829 | s = qemu_mallocz(sizeof(SerialState)); |
| 525 | if (!s) | 830 | if (!s) |
| 526 | return NULL; | 831 | return NULL; |
| 527 | - s->irq = irq; | 832 | + |
| 528 | s->base = base; | 833 | s->base = base; |
| 529 | s->it_shift = it_shift; | 834 | s->it_shift = it_shift; |
| 530 | - s->baudbase= baudbase; | ||
| 531 | 835 | ||
| 532 | - s->tx_timer = qemu_new_timer(vm_clock, serial_tx_done, s); | ||
| 533 | - if (!s->tx_timer) | ||
| 534 | - return NULL; | ||
| 535 | - | ||
| 536 | - qemu_register_reset(serial_reset, s); | ||
| 537 | - serial_reset(s); | ||
| 538 | - | ||
| 539 | - register_savevm("serial", base, 2, serial_save, serial_load, s); | 836 | + serial_init_core(s, irq, baudbase, chr); |
| 837 | + register_savevm("serial", base, 3, serial_save, serial_load, s); | ||
| 540 | 838 | ||
| 541 | if (ioregister) { | 839 | if (ioregister) { |
| 542 | s_io_memory = cpu_register_io_memory(0, serial_mm_read, | 840 | s_io_memory = cpu_register_io_memory(0, serial_mm_read, |
| 543 | serial_mm_write, s); | 841 | serial_mm_write, s); |
| 544 | cpu_register_physical_memory(base, 8 << it_shift, s_io_memory); | 842 | cpu_register_physical_memory(base, 8 << it_shift, s_io_memory); |
| 545 | } | 843 | } |
| 546 | - s->chr = chr; | ||
| 547 | qemu_chr_add_handlers(chr, serial_can_receive1, serial_receive1, | 844 | qemu_chr_add_handlers(chr, serial_can_receive1, serial_receive1, |
| 548 | serial_event, s); | 845 | serial_event, s); |
| 846 | + serial_update_msl(s); | ||
| 549 | return s; | 847 | return s; |
| 550 | } | 848 | } |
qemu-char.h
| @@ -28,6 +28,16 @@ typedef struct { | @@ -28,6 +28,16 @@ typedef struct { | ||
| 28 | #define CHR_IOCTL_PP_EPP_WRITE_ADDR 10 | 28 | #define CHR_IOCTL_PP_EPP_WRITE_ADDR 10 |
| 29 | #define CHR_IOCTL_PP_EPP_WRITE 11 | 29 | #define CHR_IOCTL_PP_EPP_WRITE 11 |
| 30 | 30 | ||
| 31 | +#define CHR_IOCTL_SERIAL_SET_TIOCM 12 | ||
| 32 | +#define CHR_IOCTL_SERIAL_GET_TIOCM 13 | ||
| 33 | + | ||
| 34 | +#define CHR_TIOCM_CTS 0x020 | ||
| 35 | +#define CHR_TIOCM_CAR 0x040 | ||
| 36 | +#define CHR_TIOCM_DSR 0x100 | ||
| 37 | +#define CHR_TIOCM_RI 0x080 | ||
| 38 | +#define CHR_TIOCM_DTR 0x002 | ||
| 39 | +#define CHR_TIOCM_RTS 0x004 | ||
| 40 | + | ||
| 31 | typedef void IOEventHandler(void *opaque, int event); | 41 | typedef void IOEventHandler(void *opaque, int event); |
| 32 | 42 | ||
| 33 | struct CharDriverState { | 43 | struct CharDriverState { |
vl.c
| @@ -2721,6 +2721,37 @@ static int tty_serial_ioctl(CharDriverState *chr, int cmd, void *arg) | @@ -2721,6 +2721,37 @@ static int tty_serial_ioctl(CharDriverState *chr, int cmd, void *arg) | ||
| 2721 | tcsendbreak(s->fd_in, 1); | 2721 | tcsendbreak(s->fd_in, 1); |
| 2722 | } | 2722 | } |
| 2723 | break; | 2723 | break; |
| 2724 | + case CHR_IOCTL_SERIAL_GET_TIOCM: | ||
| 2725 | + { | ||
| 2726 | + int sarg = 0; | ||
| 2727 | + int *targ = (int *)arg; | ||
| 2728 | + ioctl(s->fd_in, TIOCMGET, &sarg); | ||
| 2729 | + *targ = 0; | ||
| 2730 | + if (sarg | TIOCM_CTS) | ||
| 2731 | + *targ |= CHR_TIOCM_CTS; | ||
| 2732 | + if (sarg | TIOCM_CAR) | ||
| 2733 | + *targ |= CHR_TIOCM_CAR; | ||
| 2734 | + if (sarg | TIOCM_DSR) | ||
| 2735 | + *targ |= CHR_TIOCM_DSR; | ||
| 2736 | + if (sarg | TIOCM_RI) | ||
| 2737 | + *targ |= CHR_TIOCM_RI; | ||
| 2738 | + if (sarg | TIOCM_DTR) | ||
| 2739 | + *targ |= CHR_TIOCM_DTR; | ||
| 2740 | + if (sarg | TIOCM_RTS) | ||
| 2741 | + *targ |= CHR_TIOCM_RTS; | ||
| 2742 | + } | ||
| 2743 | + break; | ||
| 2744 | + case CHR_IOCTL_SERIAL_SET_TIOCM: | ||
| 2745 | + { | ||
| 2746 | + int sarg = *(int *)arg; | ||
| 2747 | + int targ = 0; | ||
| 2748 | + if (sarg | CHR_TIOCM_DTR) | ||
| 2749 | + targ |= TIOCM_DTR; | ||
| 2750 | + if (sarg | CHR_TIOCM_RTS) | ||
| 2751 | + targ |= TIOCM_RTS; | ||
| 2752 | + ioctl(s->fd_in, TIOCMSET, &targ); | ||
| 2753 | + } | ||
| 2754 | + break; | ||
| 2724 | default: | 2755 | default: |
| 2725 | return -ENOTSUP; | 2756 | return -ENOTSUP; |
| 2726 | } | 2757 | } |