Sync with dev/ic/com.c rev 1.221.

This commit is contained in:
simonb 2003-11-07 02:08:35 +00:00
parent cd9bd33872
commit 4116da8027

View File

@ -1,6 +1,7 @@
#define AU1x00_UART /* XXX */ #define AU1x00_UART /* XXX */
/* $NetBSD: aucom.c,v 1.10 2003/08/07 16:28:25 agc Exp $ */ /* $NetBSD: aucom.c,v 1.11 2003/11/07 02:08:35 simonb Exp $ */
/* NetBSD: com.c,v 1.221 2003/11/06 23:02:27 simonb Exp */
/*- /*-
* Copyright (c) 1998, 1999 The NetBSD Foundation, Inc. * Copyright (c) 1998, 1999 The NetBSD Foundation, Inc.
@ -76,7 +77,7 @@
* XXX: hacked to work with almost 16550-alike Alchemy Au1X00 on-chip uarts * XXX: hacked to work with almost 16550-alike Alchemy Au1X00 on-chip uarts
*/ */
#include <sys/cdefs.h> #include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: aucom.c,v 1.10 2003/08/07 16:28:25 agc Exp $"); __KERNEL_RCSID(0, "$NetBSD: aucom.c,v 1.11 2003/11/07 02:08:35 simonb Exp $");
#include "opt_com.h" #include "opt_com.h"
#include "opt_ddb.h" #include "opt_ddb.h"
@ -188,7 +189,7 @@ static void com_enable_debugport(struct com_softc *);
void com_config(struct com_softc *); void com_config(struct com_softc *);
void com_shutdown(struct com_softc *); void com_shutdown(struct com_softc *);
int comspeed(long, long); int comspeed(long, long, int);
static u_char cflag2lcr(tcflag_t); static u_char cflag2lcr(tcflag_t);
int comparam(struct tty *, struct termios *); int comparam(struct tty *, struct termios *);
void comstart(struct tty *); void comstart(struct tty *);
@ -205,7 +206,7 @@ void com_iflush(struct com_softc *);
int com_common_getc(dev_t, bus_space_tag_t, bus_space_handle_t); int com_common_getc(dev_t, bus_space_tag_t, bus_space_handle_t);
void com_common_putc(dev_t, bus_space_tag_t, bus_space_handle_t, int); void com_common_putc(dev_t, bus_space_tag_t, bus_space_handle_t, int);
int cominit(bus_space_tag_t, bus_addr_t, int, int, tcflag_t, int cominit(bus_space_tag_t, bus_addr_t, int, int, int, tcflag_t,
bus_space_handle_t *); bus_space_handle_t *);
int comcngetc(dev_t); int comcngetc(dev_t);
@ -315,8 +316,9 @@ void com_kgdb_putc(void *, int);
#endif #endif
/*ARGSUSED*/
int int
comspeed(long speed, long frequency) comspeed(long speed, long frequency, int type)
{ {
#define divrnd(n, q) (((n)*2/(q)+1)/2) /* divide and round off */ #define divrnd(n, q) (((n)*2/(q)+1)/2) /* divide and round off */
@ -350,20 +352,20 @@ comstatus(struct com_softc *sc, char *str)
{ {
struct tty *tp = sc->sc_tty; struct tty *tp = sc->sc_tty;
printf("%s: %s %sclocal %sdcd %sts_carr_on %sdtr %stx_stopped\n", printf("%s: %s %cclocal %cdcd %cts_carr_on %cdtr %ctx_stopped\n",
sc->sc_dev.dv_xname, str, sc->sc_dev.dv_xname, str,
ISSET(tp->t_cflag, CLOCAL) ? "+" : "-", ISSET(tp->t_cflag, CLOCAL) ? '+' : '-',
ISSET(sc->sc_msr, MSR_DCD) ? "+" : "-", ISSET(sc->sc_msr, MSR_DCD) ? '+' : '-',
ISSET(tp->t_state, TS_CARR_ON) ? "+" : "-", ISSET(tp->t_state, TS_CARR_ON) ? '+' : '-',
ISSET(sc->sc_mcr, MCR_DTR) ? "+" : "-", ISSET(sc->sc_mcr, MCR_DTR) ? '+' : '-',
sc->sc_tx_stopped ? "+" : "-"); sc->sc_tx_stopped ? '+' : '-');
printf("%s: %s %scrtscts %scts %sts_ttstop %srts %xrx_flags\n", printf("%s: %s %ccrtscts %ccts %cts_ttstop %crts rx_flags=0x%x\n",
sc->sc_dev.dv_xname, str, sc->sc_dev.dv_xname, str,
ISSET(tp->t_cflag, CRTSCTS) ? "+" : "-", ISSET(tp->t_cflag, CRTSCTS) ? '+' : '-',
ISSET(sc->sc_msr, MSR_CTS) ? "+" : "-", ISSET(sc->sc_msr, MSR_CTS) ? '+' : '-',
ISSET(tp->t_state, TS_TTSTOP) ? "+" : "-", ISSET(tp->t_state, TS_TTSTOP) ? '+' : '-',
ISSET(sc->sc_mcr, MCR_RTS) ? "+" : "-", ISSET(sc->sc_mcr, MCR_RTS) ? '+' : '-',
sc->sc_rx_flags); sc->sc_rx_flags);
} }
#endif #endif
@ -439,7 +441,7 @@ comprobeHAYESP(bus_space_handle_t hayespioh, struct com_softc *sc)
* better), at the correct com port address. * better), at the correct com port address.
*/ */
SET(sc->sc_hwflags, COM_HW_HAYESP); sc->sc_type = COM_TYPE_HAYESP;
printf(", 1024 byte fifo\n"); printf(", 1024 byte fifo\n");
return (1); return (1);
} }
@ -454,6 +456,10 @@ com_enable_debugport(struct com_softc *sc)
s = splserial(); s = splserial();
COM_LOCK(sc); COM_LOCK(sc);
sc->sc_ier = IER_ERXRDY; sc->sc_ier = IER_ERXRDY;
#ifdef COM_PXA2X0
if (sc->sc_type == COM_TYPE_PXA2x0)
sc->sc_ier |= IER_EUART | IER_ERXTOUT;
#endif
bus_space_write_1(sc->sc_iot, sc->sc_ioh, com_ier, sc->sc_ier); bus_space_write_1(sc->sc_iot, sc->sc_ioh, com_ier, sc->sc_ier);
SET(sc->sc_mcr, MCR_DTR | MCR_RTS); SET(sc->sc_mcr, MCR_DTR | MCR_RTS);
bus_space_write_1(sc->sc_iot, sc->sc_ioh, com_mcr, sc->sc_mcr); bus_space_write_1(sc->sc_iot, sc->sc_ioh, com_mcr, sc->sc_mcr);
@ -475,6 +481,7 @@ com_attach_subr(struct com_softc *sc)
int hayesp_ports[] = { 0x140, 0x180, 0x280, 0x300, 0 }; int hayesp_ports[] = { 0x140, 0x180, 0x280, 0x300, 0 };
int *hayespp; int *hayespp;
#endif #endif
const char *fifo_msg = NULL;
callout_init(&sc->sc_diag_callout); callout_init(&sc->sc_diag_callout);
#if (defined(MULTIPROCESSOR) || defined(LOCKDEBUG)) && defined(COM_MPLOCK) #if (defined(MULTIPROCESSOR) || defined(LOCKDEBUG)) && defined(COM_MPLOCK)
@ -482,6 +489,11 @@ com_attach_subr(struct com_softc *sc)
#endif #endif
/* Disable interrupts before configuring the device. */ /* Disable interrupts before configuring the device. */
#ifdef COM_PXA2X0
if (sc->sc_type == COM_TYPE_PXA2x0)
sc->sc_ier = IER_EUART;
else
#endif
sc->sc_ier = 0; sc->sc_ier = 0;
bus_space_write_1(iot, ioh, com_ier, sc->sc_ier); bus_space_write_1(iot, ioh, com_ier, sc->sc_ier);
@ -513,10 +525,10 @@ com_attach_subr(struct com_softc *sc)
bus_space_unmap(iot, hayespioh, HAYESP_NPORTS); bus_space_unmap(iot, hayespioh, HAYESP_NPORTS);
} }
/* No ESP; look for other things. */ /* No ESP; look for other things. */
if (!ISSET(sc->sc_hwflags, COM_HW_HAYESP)) { if (sc->sc_type != COM_TYPE_HAYESP) {
#endif #endif
#ifdef AU1x00_UART #ifdef AU1x00_UART
printf(": Au1X00 UART\n"); fifo_msg = "Au1X00 UART, working fifo";
sc->sc_fifolen = 16; sc->sc_fifolen = 16;
SET(sc->sc_hwflags, COM_HW_FIFO); SET(sc->sc_hwflags, COM_HW_FIFO);
#else /* !AU1x00_UART */ #else /* !AU1x00_UART */
@ -563,21 +575,29 @@ com_attach_subr(struct com_softc *sc)
#ifdef COM16650 #ifdef COM16650
bus_space_write_1(iot, ioh, com_lcr, lcr); bus_space_write_1(iot, ioh, com_lcr, lcr);
if (sc->sc_fifolen == 0) if (sc->sc_fifolen == 0)
printf(": st16650, broken fifo\n"); fifo_msg = "st16650, broken fifo";
else if (sc->sc_fifolen == 32) else if (sc->sc_fifolen == 32)
printf(": st16650a, working fifo\n"); fifo_msg = "st16650a, working fifo";
else else
#endif #endif
printf(": ns16550a, working fifo\n"); fifo_msg = "ns16550a, working fifo";
} else } else
printf(": ns16550, broken fifo\n"); fifo_msg = "ns16550, broken fifo";
else else
printf(": ns8250 or ns16450, no fifo\n"); fifo_msg = "ns8250 or ns16450, no fifo";
#endif /* !AU1x00_UART */ #endif /* !AU1x00_UART */
bus_space_write_1(iot, ioh, com_fifo, 0); bus_space_write_1(iot, ioh, com_fifo, 0);
/*
* Some chips will clear down both Tx and Rx FIFOs when zero is
* written to com_fifo. If this chip is the console, writing zero
* results in some of the chip/FIFO description being lost, so delay
* printing it until now.
*/
delay(10);
aprint_normal(": %s\n", fifo_msg);
if (ISSET(sc->sc_hwflags, COM_HW_TXFIFO_DISABLE)) { if (ISSET(sc->sc_hwflags, COM_HW_TXFIFO_DISABLE)) {
sc->sc_fifolen = 1; sc->sc_fifolen = 1;
printf("%s: txfifo disabled\n", sc->sc_dev.dv_xname); aprint_normal("%s: txfifo disabled\n", sc->sc_dev.dv_xname);
} }
#ifdef COM_HAYESP #ifdef COM_HAYESP
} }
@ -593,7 +613,7 @@ com_attach_subr(struct com_softc *sc)
sc->sc_rbput = sc->sc_rbget = sc->sc_rbuf; sc->sc_rbput = sc->sc_rbget = sc->sc_rbuf;
sc->sc_rbavail = com_rbuf_size; sc->sc_rbavail = com_rbuf_size;
if (sc->sc_rbuf == NULL) { if (sc->sc_rbuf == NULL) {
printf("%s: unable to allocate ring buffer\n", aprint_error("%s: unable to allocate ring buffer\n",
sc->sc_dev.dv_xname); sc->sc_dev.dv_xname);
return; return;
} }
@ -610,21 +630,25 @@ com_attach_subr(struct com_softc *sc)
/* locate the major number */ /* locate the major number */
maj = cdevsw_lookup_major(&com_cdevsw); maj = cdevsw_lookup_major(&com_cdevsw);
cn_tab->cn_dev = makedev(maj, sc->sc_dev.dv_unit); tp->t_dev = cn_tab->cn_dev = makedev(maj, sc->sc_dev.dv_unit);
printf("%s: console\n", sc->sc_dev.dv_xname); aprint_normal("%s: console\n", sc->sc_dev.dv_xname);
} }
#ifdef KGDB #ifdef KGDB
/* /*
* Allow kgdb to "take over" this port. If this is * Allow kgdb to "take over" this port. If this is
* the kgdb device, it has exclusive use. * not the console and is the kgdb device, it has
* exclusive use. If it's the console _and_ the
* kgdb device, it doesn't.
*/ */
if (iot == com_kgdb_iot && iobase == com_kgdb_addr) { if (iot == com_kgdb_iot && iobase == com_kgdb_addr) {
if (!ISSET(sc->sc_hwflags, COM_HW_CONSOLE)) {
com_kgdb_attached = 1; com_kgdb_attached = 1;
SET(sc->sc_hwflags, COM_HW_KGDB); SET(sc->sc_hwflags, COM_HW_KGDB);
printf("%s: kgdb\n", sc->sc_dev.dv_xname); }
aprint_normal("%s: kgdb\n", sc->sc_dev.dv_xname);
} }
#endif #endif
@ -654,12 +678,17 @@ com_config(struct com_softc *sc)
bus_space_handle_t ioh = sc->sc_ioh; bus_space_handle_t ioh = sc->sc_ioh;
/* Disable interrupts before configuring the device. */ /* Disable interrupts before configuring the device. */
#ifdef COM_PXA2X0
if (sc->sc_type == COM_TYPE_PXA2x0)
sc->sc_ier = IER_EUART;
else
#endif
sc->sc_ier = 0; sc->sc_ier = 0;
bus_space_write_1(iot, ioh, com_ier, sc->sc_ier); bus_space_write_1(iot, ioh, com_ier, sc->sc_ier);
#ifdef COM_HAYESP #ifdef COM_HAYESP
/* Look for a Hayes ESP board. */ /* Look for a Hayes ESP board. */
if (ISSET(sc->sc_hwflags, COM_HW_HAYESP)) { if (sc->sc_type == COM_TYPE_HAYESP) {
sc->sc_fifolen = 1024; sc->sc_fifolen = 1024;
/* Set 16550 compatibility mode */ /* Set 16550 compatibility mode */
@ -807,10 +836,20 @@ com_shutdown(struct com_softc *sc)
} }
/* Turn off interrupts. */ /* Turn off interrupts. */
if (ISSET(sc->sc_hwflags, COM_HW_CONSOLE)) if (ISSET(sc->sc_hwflags, COM_HW_CONSOLE)) {
sc->sc_ier = IER_ERXRDY; /* interrupt on break */ sc->sc_ier = IER_ERXRDY; /* interrupt on break */
else #ifdef COM_PXA2X0
if (sc->sc_type == COM_TYPE_PXA2x0)
sc->sc_ier |= IER_ERXTOUT;
#endif
} else
sc->sc_ier = 0; sc->sc_ier = 0;
#ifdef COM_PXA2X0
if (sc->sc_type == COM_TYPE_PXA2x0)
sc->sc_ier |= IER_EUART;
#endif
bus_space_write_1(sc->sc_iot, sc->sc_ioh, com_ier, sc->sc_ier); bus_space_write_1(sc->sc_iot, sc->sc_ioh, com_ier, sc->sc_ier);
if (sc->disable) { if (sc->disable) {
@ -884,6 +923,10 @@ comopen(dev_t dev, int flag, int mode, struct proc *p)
/* Turn on interrupts. */ /* Turn on interrupts. */
sc->sc_ier = IER_ERXRDY | IER_ERLS | IER_EMSC; sc->sc_ier = IER_ERXRDY | IER_ERLS | IER_EMSC;
#ifdef COM_PXA2X0
if (sc->sc_type == COM_TYPE_PXA2x0)
sc->sc_ier |= IER_EUART | IER_ERXTOUT;
#endif
bus_space_write_1(sc->sc_iot, sc->sc_ioh, com_ier, sc->sc_ier); bus_space_write_1(sc->sc_iot, sc->sc_ioh, com_ier, sc->sc_ier);
/* Fetch the current modem control status, needed later. */ /* Fetch the current modem control status, needed later. */
@ -1200,7 +1243,7 @@ comioctl(dev_t dev, u_long cmd, caddr_t data, int flag, struct proc *p)
TIMESPEC_TO_TIMEVAL((struct timeval *)data, TIMESPEC_TO_TIMEVAL((struct timeval *)data,
&sc->ppsinfo.assert_timestamp); &sc->ppsinfo.assert_timestamp);
#else #else
sc->sc_ppsassert = -1 sc->sc_ppsassert = -1;
sc->sc_ppsclear = 0; sc->sc_ppsclear = 0;
TIMESPEC_TO_TIMEVAL((struct timeval *)data, TIMESPEC_TO_TIMEVAL((struct timeval *)data,
&sc->ppsinfo.clear_timestamp); &sc->ppsinfo.clear_timestamp);
@ -1343,7 +1386,13 @@ com_to_tiocm(struct com_softc *sc)
if (ISSET(combits, MSR_RI | MSR_TERI)) if (ISSET(combits, MSR_RI | MSR_TERI))
SET(ttybits, TIOCM_RI); SET(ttybits, TIOCM_RI);
if (sc->sc_ier != 0) #ifdef COM_PXA2X0
if (sc->sc_type == COM_TYPE_PXA2x0) {
if ((sc->sc_ier & 0x0f) != 0)
SET(ttybits, TIOCM_LE);
} else
#endif
if ((sc->sc_ier & 0xbf) != 0)
SET(ttybits, TIOCM_LE); SET(ttybits, TIOCM_LE);
return (ttybits); return (ttybits);
@ -1391,7 +1440,7 @@ comparam(struct tty *tp, struct termios *t)
return (EIO); return (EIO);
#ifdef COM_HAYESP #ifdef COM_HAYESP
if (ISSET(sc->sc_hwflags, COM_HW_HAYESP)) { if (sc->sc_type == COM_TYPE_HAYESP) {
int prescaler, speed; int prescaler, speed;
/* /*
@ -1400,7 +1449,8 @@ comparam(struct tty *tp, struct termios *t)
*/ */
for (prescaler = 0, speed = t->c_ospeed; prescaler < 4; for (prescaler = 0, speed = t->c_ospeed; prescaler < 4;
prescaler++, speed /= 2) prescaler++, speed /= 2)
if ((ospeed = comspeed(speed, sc->sc_frequency)) > 0) if ((ospeed = comspeed(speed, sc->sc_frequency,
sc->sc_type)) > 0)
break; break;
if (prescaler == 4) if (prescaler == 4)
@ -1408,7 +1458,7 @@ comparam(struct tty *tp, struct termios *t)
sc->sc_prescaler = prescaler; sc->sc_prescaler = prescaler;
} else } else
#endif #endif
ospeed = comspeed(t->c_ospeed, sc->sc_frequency); ospeed = comspeed(t->c_ospeed, sc->sc_frequency, sc->sc_type);
/* Check requested parameters. */ /* Check requested parameters. */
if (ospeed < 0) if (ospeed < 0)
@ -1511,7 +1561,7 @@ comparam(struct tty *tp, struct termios *t)
* * Otherwise set it a bit higher. * * Otherwise set it a bit higher.
*/ */
#ifdef COM_HAYESP #ifdef COM_HAYESP
if (ISSET(sc->sc_hwflags, COM_HW_HAYESP)) if (sc->sc_type == COM_TYPE_HAYESP)
sc->sc_fifo = FIFO_DMA_MODE | FIFO_ENABLE | FIFO_TRIGGER_8; sc->sc_fifo = FIFO_DMA_MODE | FIFO_ENABLE | FIFO_TRIGGER_8;
else else
#endif #endif
@ -1617,6 +1667,11 @@ com_loadchannelregs(struct com_softc *sc)
/* XXXXX necessary? */ /* XXXXX necessary? */
com_iflush(sc); com_iflush(sc);
#ifdef COM_PXA2X0
if (sc->sc_type == COM_TYPE_PXA2x0)
bus_space_write_1(iot, ioh, com_ier, IER_EUART);
else
#endif
bus_space_write_1(iot, ioh, com_ier, 0); bus_space_write_1(iot, ioh, com_ier, 0);
#ifdef com_efr /* XXX: no com_efr on Au1X00 */ #ifdef com_efr /* XXX: no com_efr on Au1X00 */
@ -1636,7 +1691,7 @@ com_loadchannelregs(struct com_softc *sc)
bus_space_write_1(iot, ioh, com_mcr, sc->sc_mcr_active = sc->sc_mcr); bus_space_write_1(iot, ioh, com_mcr, sc->sc_mcr_active = sc->sc_mcr);
bus_space_write_1(iot, ioh, com_fifo, sc->sc_fifo); bus_space_write_1(iot, ioh, com_fifo, sc->sc_fifo);
#ifdef COM_HAYESP #ifdef COM_HAYESP
if (ISSET(sc->sc_hwflags, COM_HW_HAYESP)) { if (sc->sc_type == COM_TYPE_HAYESP) {
bus_space_write_1(iot, sc->sc_hayespioh, HAYESP_CMD1, bus_space_write_1(iot, sc->sc_hayespioh, HAYESP_CMD1,
HAYESP_SETPRESCALER); HAYESP_SETPRESCALER);
bus_space_write_1(iot, sc->sc_hayespioh, HAYESP_CMD2, bus_space_write_1(iot, sc->sc_hayespioh, HAYESP_CMD2,
@ -1759,7 +1814,7 @@ comstart(struct tty *tp)
/* Output the first chunk of the contiguous buffer. */ /* Output the first chunk of the contiguous buffer. */
if (!ISSET(sc->sc_hwflags, COM_HW_NO_TXPRELOAD)) { if (!ISSET(sc->sc_hwflags, COM_HW_NO_TXPRELOAD)) {
int n; u_int n;
n = sc->sc_tbc; n = sc->sc_tbc;
if (n > sc->sc_fifolen) if (n > sc->sc_fifolen)
@ -1840,6 +1895,13 @@ com_rxsoft(struct com_softc *sc, struct tty *tp)
comdiag, sc); comdiag, sc);
} }
/* If not yet open, drop the entire buffer content here */
if (!ISSET(tp->t_state, TS_ISOPEN)) {
get += cc << 1;
if (get >= end)
get -= com_rbuf_size << 1;
cc = 0;
}
while (cc) { while (cc) {
code = get[0]; code = get[0];
lsr = get[1]; lsr = get[1];
@ -1900,7 +1962,12 @@ com_rxsoft(struct com_softc *sc, struct tty *tp)
if (ISSET(sc->sc_rx_flags, RX_IBUF_OVERFLOWED)) { if (ISSET(sc->sc_rx_flags, RX_IBUF_OVERFLOWED)) {
CLR(sc->sc_rx_flags, RX_IBUF_OVERFLOWED); CLR(sc->sc_rx_flags, RX_IBUF_OVERFLOWED);
SET(sc->sc_ier, IER_ERXRDY); SET(sc->sc_ier, IER_ERXRDY);
bus_space_write_1(sc->sc_iot, sc->sc_ioh, com_ier, sc->sc_ier); #ifdef COM_PXA2X0
if (sc->sc_type == COM_TYPE_PXA2x0)
SET(sc->sc_ier, IER_ERXTOUT);
#endif
bus_space_write_1(sc->sc_iot, sc->sc_ioh,
com_ier, sc->sc_ier);
} }
if (ISSET(sc->sc_rx_flags, RX_IBUF_BLOCKED)) { if (ISSET(sc->sc_rx_flags, RX_IBUF_BLOCKED)) {
CLR(sc->sc_rx_flags, RX_IBUF_BLOCKED); CLR(sc->sc_rx_flags, RX_IBUF_BLOCKED);
@ -2064,11 +2131,12 @@ again: do {
lsr = bus_space_read_1(iot, ioh, com_lsr); lsr = bus_space_read_1(iot, ioh, com_lsr);
if (ISSET(lsr, LSR_BI)) { if (ISSET(lsr, LSR_BI)) {
int cn_trapped = 0; int cn_trapped = 0;
cn_check_magic(sc->sc_tty->t_dev, cn_check_magic(sc->sc_tty->t_dev,
CNC_BREAK, com_cnm_state); CNC_BREAK, com_cnm_state);
if (cn_trapped) if (cn_trapped)
continue; continue;
#if defined(KGDB) #if defined(KGDB) && !defined(DDB)
if (ISSET(sc->sc_hwflags, COM_HW_KGDB)) { if (ISSET(sc->sc_hwflags, COM_HW_KGDB)) {
kgdb_connect(1); kgdb_connect(1);
continue; continue;
@ -2129,10 +2197,21 @@ again: do {
if (!cc) { if (!cc) {
SET(sc->sc_rx_flags, RX_IBUF_OVERFLOWED); SET(sc->sc_rx_flags, RX_IBUF_OVERFLOWED);
CLR(sc->sc_ier, IER_ERXRDY); CLR(sc->sc_ier, IER_ERXRDY);
bus_space_write_1(iot, ioh, com_ier, sc->sc_ier); #ifdef COM_PXA2X0
if (sc->sc_type == COM_TYPE_PXA2x0)
CLR(sc->sc_ier, IER_ERXTOUT);
#endif
bus_space_write_1(iot, ioh, com_ier,
sc->sc_ier);
} }
} else { } else {
if ((iir & IIR_IMASK) == IIR_RXRDY) { if ((iir & IIR_IMASK) == IIR_RXRDY) {
#ifdef COM_PXA2X0
if (sc->sc_type == COM_TYPE_PXA2x0)
bus_space_write_1(iot, ioh, com_ier,
IER_EUART);
else
#endif
bus_space_write_1(iot, ioh, com_ier, 0); bus_space_write_1(iot, ioh, com_ier, 0);
delay(10); delay(10);
bus_space_write_1(iot, ioh, com_ier,sc->sc_ier); bus_space_write_1(iot, ioh, com_ier,sc->sc_ier);
@ -2230,7 +2309,7 @@ again: do {
/* Output the next chunk of the contiguous buffer, if any. */ /* Output the next chunk of the contiguous buffer, if any. */
if (sc->sc_tbc > 0) { if (sc->sc_tbc > 0) {
int n; u_int n;
n = sc->sc_tbc; n = sc->sc_tbc;
if (n > sc->sc_fifolen) if (n > sc->sc_fifolen)
@ -2331,9 +2410,8 @@ void
com_common_putc(dev_t dev, bus_space_tag_t iot, bus_space_handle_t ioh, int c) com_common_putc(dev_t dev, bus_space_tag_t iot, bus_space_handle_t ioh, int c)
{ {
int s = splserial(); int s = splserial();
int timo; int cin, stat, timo;
int cin, stat;
if (com_readaheadcount < MAX_READAHEAD if (com_readaheadcount < MAX_READAHEAD
&& ISSET(stat = bus_space_read_1(iot, ioh, com_lsr), LSR_RXRDY)) { && ISSET(stat = bus_space_read_1(iot, ioh, com_lsr), LSR_RXRDY)) {
int cn_trapped = 0; int cn_trapped = 0;
@ -2364,7 +2442,7 @@ com_common_putc(dev_t dev, bus_space_tag_t iot, bus_space_handle_t ioh, int c)
*/ */
int int
cominit(bus_space_tag_t iot, bus_addr_t iobase, int rate, int frequency, cominit(bus_space_tag_t iot, bus_addr_t iobase, int rate, int frequency,
tcflag_t cflag, bus_space_handle_t *iohp) int type, tcflag_t cflag, bus_space_handle_t *iohp)
{ {
bus_space_handle_t ioh; bus_space_handle_t ioh;
@ -2376,7 +2454,7 @@ cominit(bus_space_tag_t iot, bus_addr_t iobase, int rate, int frequency,
bus_space_write_1(iot, ioh, com_efr, 0); bus_space_write_1(iot, ioh, com_efr, 0);
bus_space_write_1(iot, ioh, com_lcr, LCR_DLAB); bus_space_write_1(iot, ioh, com_lcr, LCR_DLAB);
#endif #endif
rate = comspeed(rate, frequency); rate = comspeed(rate, frequency, type);
#ifdef com_dlb /* XXXsimonb: Au1X00 has single clock divisor register */ #ifdef com_dlb /* XXXsimonb: Au1X00 has single clock divisor register */
bus_space_write_2(iot, ioh, com_dlb, rate); bus_space_write_2(iot, ioh, com_dlb, rate);
#else #else
@ -2387,6 +2465,11 @@ cominit(bus_space_tag_t iot, bus_addr_t iobase, int rate, int frequency,
bus_space_write_1(iot, ioh, com_mcr, MCR_DTR | MCR_RTS); bus_space_write_1(iot, ioh, com_mcr, MCR_DTR | MCR_RTS);
bus_space_write_1(iot, ioh, com_fifo, bus_space_write_1(iot, ioh, com_fifo,
FIFO_ENABLE | FIFO_RCV_RST | FIFO_XMT_RST | FIFO_TRIGGER_1); FIFO_ENABLE | FIFO_RCV_RST | FIFO_XMT_RST | FIFO_TRIGGER_1);
#ifdef COM_PXA2X0
if (sc->sc_type == COM_TYPE_PXA2x0)
bus_space_write_1(iot, ioh, com_ier, IER_EUART);
else
#endif
bus_space_write_1(iot, ioh, com_ier, 0); bus_space_write_1(iot, ioh, com_ier, 0);
*iohp = ioh; *iohp = ioh;
@ -2397,19 +2480,18 @@ cominit(bus_space_tag_t iot, bus_addr_t iobase, int rate, int frequency,
* Following are all routines needed for COM to act as console * Following are all routines needed for COM to act as console
*/ */
struct consdev comcons = { struct consdev comcons = {
NULL, NULL, comcngetc, comcnputc, comcnpollc, NULL, NULL, NULL, comcngetc, comcnputc, comcnpollc, NULL, NULL, NULL,
NULL, NULL, NODEV, CN_NORMAL NODEV, CN_NORMAL
}; };
/*ARGSUSED*/
int int
comcnattach(bus_space_tag_t iot, bus_addr_t iobase, int rate, int frequency, comcnattach(bus_space_tag_t iot, bus_addr_t iobase, int rate, int frequency,
int type, tcflag_t cflag) int type, tcflag_t cflag)
{ {
int res; int res;
res = cominit(iot, iobase, rate, frequency, cflag, &comconsioh); res = cominit(iot, iobase, rate, frequency, type, cflag, &comconsioh);
if (res) if (res)
return (res); return (res);
@ -2451,14 +2533,19 @@ comcnpollc(dev_t dev, int on)
#ifdef KGDB #ifdef KGDB
int int
com_kgdb_attach(bus_space_tag_t iot, bus_addr_t iobase, int rate, com_kgdb_attach(bus_space_tag_t iot, bus_addr_t iobase, int rate,
int frequency, tcflag_t cflag) int frequency, int type, tcflag_t cflag)
{ {
int res; int res;
if (iot == comconstag && iobase == comconsaddr) if (iot == comconstag && iobase == comconsaddr) {
#if !defined(DDB)
return (EBUSY); /* cannot share with console */ return (EBUSY); /* cannot share with console */
#else
res = cominit(iot, iobase, rate, frequency, cflag, &com_kgdb_ioh); com_kgdb_ioh = comconsioh;
#endif
} else {
res = cominit(iot, iobase, rate, frequency, type, cflag,
&com_kgdb_ioh);
if (res) if (res)
return (res); return (res);
@ -2468,6 +2555,7 @@ com_kgdb_attach(bus_space_tag_t iot, bus_addr_t iobase, int rate,
*/ */
cn_init_magic(&com_cnm_state); cn_init_magic(&com_cnm_state);
cn_set_magic("\047\001"); cn_set_magic("\047\001");
}
kgdb_attach(com_kgdb_getc, com_kgdb_putc, NULL); kgdb_attach(com_kgdb_getc, com_kgdb_putc, NULL);
kgdb_dev = 123; /* unneeded, only to satisfy some tests */ kgdb_dev = 123; /* unneeded, only to satisfy some tests */