Sync with dev/ic/com.c rev 1.221.
This commit is contained in:
parent
cd9bd33872
commit
4116da8027
@ -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 */
|
||||||
|
Loading…
Reference in New Issue
Block a user