2021-05-04 18:30:07 +00:00
|
|
|
/* SPDX-License-Identifier: MIT */
|
|
|
|
|
|
|
|
#include "hv.h"
|
2021-08-23 08:02:28 +00:00
|
|
|
#include "aic.h"
|
|
|
|
#include "uart.h"
|
2021-05-04 18:30:07 +00:00
|
|
|
#include "uart_regs.h"
|
|
|
|
|
2021-05-14 20:47:43 +00:00
|
|
|
static iodev_id_t vuart_iodev;
|
|
|
|
|
2021-08-23 08:02:28 +00:00
|
|
|
bool active = false;
|
|
|
|
|
|
|
|
u32 ucon = 0;
|
|
|
|
u32 utrstat = 0;
|
|
|
|
u32 ufstat = 0;
|
|
|
|
|
|
|
|
int vuart_irq = 0;
|
|
|
|
|
|
|
|
static void update_irq(void)
|
|
|
|
{
|
|
|
|
ssize_t rx_queued;
|
|
|
|
|
|
|
|
iodev_handle_events(vuart_iodev);
|
|
|
|
|
|
|
|
utrstat |= UTRSTAT_TXBE | UTRSTAT_TXE;
|
|
|
|
utrstat &= ~UTRSTAT_RXD;
|
|
|
|
|
|
|
|
if ((rx_queued = iodev_can_read(vuart_iodev))) {
|
|
|
|
utrstat |= UTRSTAT_RXD;
|
|
|
|
if (rx_queued > 15)
|
|
|
|
ufstat = FIELD_PREP(UFSTAT_RXCNT, 15) | UFSTAT_RXFULL;
|
|
|
|
else
|
|
|
|
ufstat = FIELD_PREP(UFSTAT_RXCNT, rx_queued);
|
|
|
|
|
|
|
|
if (FIELD_GET(UCON_RXMODE, ucon) == UCON_MODE_IRQ && ucon & UCON_RXTO_ENA) {
|
|
|
|
utrstat |= UTRSTAT_RXTO;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (FIELD_GET(UCON_TXMODE, ucon) == UCON_MODE_IRQ && ucon & UCON_TXTHRESH_ENA) {
|
|
|
|
utrstat |= UTRSTAT_TXTHRESH;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vuart_irq) {
|
|
|
|
uart_clear_irqs();
|
|
|
|
if (utrstat & (UTRSTAT_TXTHRESH | UTRSTAT_RXTHRESH | UTRSTAT_RXTO)) {
|
|
|
|
aic_set_sw(vuart_irq, true);
|
|
|
|
} else {
|
|
|
|
aic_set_sw(vuart_irq, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// printf("HV: vuart UTRSTAT=0x%x UFSTAT=0x%x UCON=0x%x\n", utrstat, ufstat, ucon);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool handle_vuart(u64 addr, u64 *val, bool write, int width)
|
2021-05-04 18:30:07 +00:00
|
|
|
{
|
|
|
|
UNUSED(width);
|
2021-08-23 08:02:28 +00:00
|
|
|
|
2021-05-04 18:30:07 +00:00
|
|
|
addr &= 0xfff;
|
|
|
|
|
2021-08-23 08:02:28 +00:00
|
|
|
update_irq();
|
|
|
|
|
2021-05-04 18:30:07 +00:00
|
|
|
if (write) {
|
2021-08-23 08:02:28 +00:00
|
|
|
// printf("HV: vuart W 0x%lx <- 0x%lx (%d)\n", addr, *val, width);
|
2021-05-04 18:30:07 +00:00
|
|
|
switch (addr) {
|
2021-08-23 08:02:28 +00:00
|
|
|
case UCON:
|
|
|
|
ucon = *val;
|
|
|
|
break;
|
2021-05-04 18:30:07 +00:00
|
|
|
case UTXH: {
|
|
|
|
uint8_t b = *val;
|
2021-05-14 20:47:43 +00:00
|
|
|
if (iodev_can_write(vuart_iodev))
|
|
|
|
iodev_write(vuart_iodev, &b, 1);
|
2021-05-04 18:30:07 +00:00
|
|
|
break;
|
|
|
|
}
|
2021-08-23 08:02:28 +00:00
|
|
|
case UTRSTAT:
|
|
|
|
utrstat &= ~(*val & (UTRSTAT_TXTHRESH | UTRSTAT_RXTHRESH | UTRSTAT_RXTO));
|
|
|
|
break;
|
2021-05-04 18:30:07 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
switch (addr) {
|
2021-08-23 08:02:28 +00:00
|
|
|
case UCON:
|
|
|
|
*val = ucon;
|
|
|
|
break;
|
|
|
|
case URXH:
|
|
|
|
if (iodev_can_read(vuart_iodev)) {
|
|
|
|
uint8_t c;
|
|
|
|
iodev_read(vuart_iodev, &c, 1);
|
|
|
|
*val = c;
|
|
|
|
} else {
|
|
|
|
*val = 0;
|
|
|
|
}
|
|
|
|
break;
|
2021-05-04 18:30:07 +00:00
|
|
|
case UTRSTAT:
|
2021-08-23 08:02:28 +00:00
|
|
|
*val = utrstat;
|
|
|
|
break;
|
|
|
|
case UFSTAT:
|
|
|
|
*val = ufstat;
|
2021-05-04 18:30:07 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
*val = 0;
|
|
|
|
break;
|
|
|
|
}
|
2021-08-23 08:02:28 +00:00
|
|
|
// printf("HV: vuart R 0x%lx -> 0x%lx (%d)\n", addr, *val, width);
|
2021-05-04 18:30:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-08-23 08:02:28 +00:00
|
|
|
void hv_vuart_poll(void)
|
|
|
|
{
|
|
|
|
if (!active)
|
|
|
|
return;
|
|
|
|
|
|
|
|
update_irq();
|
|
|
|
}
|
|
|
|
|
|
|
|
void hv_map_vuart(u64 base, int irq, iodev_id_t iodev)
|
2021-05-04 18:30:07 +00:00
|
|
|
{
|
|
|
|
hv_map_hook(base, handle_vuart, 0x1000);
|
2021-05-14 20:47:43 +00:00
|
|
|
vuart_iodev = iodev;
|
2021-08-23 08:02:28 +00:00
|
|
|
vuart_irq = irq;
|
|
|
|
active = true;
|
2021-05-04 18:30:07 +00:00
|
|
|
}
|