1098ca2bdSWarner Losh /*- 227d5dc18SMarcel Moolenaar * Copyright (c) 2003 Marcel Moolenaar 327d5dc18SMarcel Moolenaar * All rights reserved. 427d5dc18SMarcel Moolenaar * 527d5dc18SMarcel Moolenaar * Redistribution and use in source and binary forms, with or without 627d5dc18SMarcel Moolenaar * modification, are permitted provided that the following conditions 727d5dc18SMarcel Moolenaar * are met: 827d5dc18SMarcel Moolenaar * 927d5dc18SMarcel Moolenaar * 1. Redistributions of source code must retain the above copyright 1027d5dc18SMarcel Moolenaar * notice, this list of conditions and the following disclaimer. 1127d5dc18SMarcel Moolenaar * 2. Redistributions in binary form must reproduce the above copyright 1227d5dc18SMarcel Moolenaar * notice, this list of conditions and the following disclaimer in the 1327d5dc18SMarcel Moolenaar * documentation and/or other materials provided with the distribution. 1427d5dc18SMarcel Moolenaar * 1527d5dc18SMarcel Moolenaar * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR 1627d5dc18SMarcel Moolenaar * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 1727d5dc18SMarcel Moolenaar * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 1827d5dc18SMarcel Moolenaar * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, 1927d5dc18SMarcel Moolenaar * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 2027d5dc18SMarcel Moolenaar * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 2127d5dc18SMarcel Moolenaar * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 2227d5dc18SMarcel Moolenaar * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 2327d5dc18SMarcel Moolenaar * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 2427d5dc18SMarcel Moolenaar * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 2527d5dc18SMarcel Moolenaar */ 2627d5dc18SMarcel Moolenaar 2727d5dc18SMarcel Moolenaar #include <sys/cdefs.h> 2827d5dc18SMarcel Moolenaar __FBSDID("$FreeBSD$"); 2927d5dc18SMarcel Moolenaar 3027d5dc18SMarcel Moolenaar #include <sys/param.h> 3127d5dc18SMarcel Moolenaar #include <sys/systm.h> 3227d5dc18SMarcel Moolenaar #include <sys/bus.h> 3327d5dc18SMarcel Moolenaar #include <sys/conf.h> 3427d5dc18SMarcel Moolenaar #include <sys/cons.h> 3527d5dc18SMarcel Moolenaar #include <sys/fcntl.h> 3627d5dc18SMarcel Moolenaar #include <sys/interrupt.h> 3727d5dc18SMarcel Moolenaar #include <sys/kernel.h> 3827d5dc18SMarcel Moolenaar #include <sys/malloc.h> 3927d5dc18SMarcel Moolenaar #include <sys/reboot.h> 4027d5dc18SMarcel Moolenaar #include <machine/bus.h> 4127d5dc18SMarcel Moolenaar #include <sys/rman.h> 4227d5dc18SMarcel Moolenaar #include <sys/tty.h> 4327d5dc18SMarcel Moolenaar #include <machine/resource.h> 4427d5dc18SMarcel Moolenaar #include <machine/stdarg.h> 4527d5dc18SMarcel Moolenaar 4627d5dc18SMarcel Moolenaar #include <dev/uart/uart.h> 4727d5dc18SMarcel Moolenaar #include <dev/uart/uart_bus.h> 4827d5dc18SMarcel Moolenaar #include <dev/uart/uart_cpu.h> 4927d5dc18SMarcel Moolenaar 5027d5dc18SMarcel Moolenaar #include "uart_if.h" 5127d5dc18SMarcel Moolenaar 5227d5dc18SMarcel Moolenaar static cn_probe_t uart_cnprobe; 5327d5dc18SMarcel Moolenaar static cn_init_t uart_cninit; 5427d5dc18SMarcel Moolenaar static cn_term_t uart_cnterm; 5527d5dc18SMarcel Moolenaar static cn_getc_t uart_cngetc; 5627d5dc18SMarcel Moolenaar static cn_putc_t uart_cnputc; 579976156fSAndriy Gapon static cn_grab_t uart_cngrab; 589976156fSAndriy Gapon static cn_ungrab_t uart_cnungrab; 5927d5dc18SMarcel Moolenaar 607672c959SPoul-Henning Kamp CONSOLE_DRIVER(uart); 6127d5dc18SMarcel Moolenaar 6227d5dc18SMarcel Moolenaar static struct uart_devinfo uart_console; 6327d5dc18SMarcel Moolenaar 6427d5dc18SMarcel Moolenaar static void 6527d5dc18SMarcel Moolenaar uart_cnprobe(struct consdev *cp) 6627d5dc18SMarcel Moolenaar { 6727d5dc18SMarcel Moolenaar 6827d5dc18SMarcel Moolenaar cp->cn_pri = CN_DEAD; 6927d5dc18SMarcel Moolenaar 7027d5dc18SMarcel Moolenaar KASSERT(uart_console.cookie == NULL, ("foo")); 7127d5dc18SMarcel Moolenaar 7227d5dc18SMarcel Moolenaar if (uart_cpu_getdev(UART_DEV_CONSOLE, &uart_console)) 7327d5dc18SMarcel Moolenaar return; 7427d5dc18SMarcel Moolenaar 7527d5dc18SMarcel Moolenaar if (uart_probe(&uart_console)) 7627d5dc18SMarcel Moolenaar return; 7727d5dc18SMarcel Moolenaar 789f0974f9SMarcel Moolenaar strlcpy(cp->cn_name, uart_driver_name, sizeof(cp->cn_name)); 7927d5dc18SMarcel Moolenaar cp->cn_pri = (boothowto & RB_SERIAL) ? CN_REMOTE : CN_NORMAL; 8027d5dc18SMarcel Moolenaar cp->cn_arg = &uart_console; 8127d5dc18SMarcel Moolenaar } 8227d5dc18SMarcel Moolenaar 8327d5dc18SMarcel Moolenaar static void 8427d5dc18SMarcel Moolenaar uart_cninit(struct consdev *cp) 8527d5dc18SMarcel Moolenaar { 8627d5dc18SMarcel Moolenaar struct uart_devinfo *di; 8727d5dc18SMarcel Moolenaar 8827d5dc18SMarcel Moolenaar /* 8927d5dc18SMarcel Moolenaar * Yedi trick: we need to be able to define cn_dev before we go 9027d5dc18SMarcel Moolenaar * single- or multi-user. The problem is that we don't know at 9127d5dc18SMarcel Moolenaar * this time what the device will be. Hence, we need to link from 9227d5dc18SMarcel Moolenaar * the uart_devinfo to the consdev that corresponds to it so that 9327d5dc18SMarcel Moolenaar * we can define cn_dev in uart_bus_attach() when we find the 9427d5dc18SMarcel Moolenaar * device during bus enumeration. That's when we'll know what the 9527d5dc18SMarcel Moolenaar * the unit number will be. 9627d5dc18SMarcel Moolenaar */ 9727d5dc18SMarcel Moolenaar di = cp->cn_arg; 9827d5dc18SMarcel Moolenaar KASSERT(di->cookie == NULL, ("foo")); 9927d5dc18SMarcel Moolenaar di->cookie = cp; 10027d5dc18SMarcel Moolenaar di->type = UART_DEV_CONSOLE; 10127d5dc18SMarcel Moolenaar uart_add_sysdev(di); 10227d5dc18SMarcel Moolenaar uart_init(di); 10327d5dc18SMarcel Moolenaar } 10427d5dc18SMarcel Moolenaar 10527d5dc18SMarcel Moolenaar static void 10627d5dc18SMarcel Moolenaar uart_cnterm(struct consdev *cp) 10727d5dc18SMarcel Moolenaar { 10827d5dc18SMarcel Moolenaar 10927d5dc18SMarcel Moolenaar uart_term(cp->cn_arg); 11027d5dc18SMarcel Moolenaar } 11127d5dc18SMarcel Moolenaar 11227d5dc18SMarcel Moolenaar static void 1139976156fSAndriy Gapon uart_cngrab(struct consdev *cp) 1149976156fSAndriy Gapon { 115f83ed22cSWarner Losh 116f83ed22cSWarner Losh uart_grab(cp->cn_arg); 1179976156fSAndriy Gapon } 1189976156fSAndriy Gapon 1199976156fSAndriy Gapon static void 1209976156fSAndriy Gapon uart_cnungrab(struct consdev *cp) 1219976156fSAndriy Gapon { 122f83ed22cSWarner Losh 123f83ed22cSWarner Losh uart_ungrab(cp->cn_arg); 1249976156fSAndriy Gapon } 1259976156fSAndriy Gapon 1269976156fSAndriy Gapon static void 12727d5dc18SMarcel Moolenaar uart_cnputc(struct consdev *cp, int c) 12827d5dc18SMarcel Moolenaar { 12927d5dc18SMarcel Moolenaar 13027d5dc18SMarcel Moolenaar uart_putc(cp->cn_arg, c); 13127d5dc18SMarcel Moolenaar } 13227d5dc18SMarcel Moolenaar 13327d5dc18SMarcel Moolenaar static int 13427d5dc18SMarcel Moolenaar uart_cngetc(struct consdev *cp) 13527d5dc18SMarcel Moolenaar { 13627d5dc18SMarcel Moolenaar 1377672c959SPoul-Henning Kamp return (uart_poll(cp->cn_arg)); 13827d5dc18SMarcel Moolenaar } 13927d5dc18SMarcel Moolenaar 14079a8d927SPoul-Henning Kamp static int 141bc093719SEd Schouten uart_tty_open(struct tty *tp) 14279a8d927SPoul-Henning Kamp { 14379a8d927SPoul-Henning Kamp struct uart_softc *sc; 14479a8d927SPoul-Henning Kamp 145bc093719SEd Schouten sc = tty_softc(tp); 146793bcd17SMarcel Moolenaar 147793bcd17SMarcel Moolenaar if (sc == NULL || sc->sc_leaving) 148793bcd17SMarcel Moolenaar return (ENXIO); 149793bcd17SMarcel Moolenaar 15079a8d927SPoul-Henning Kamp sc->sc_opened = 1; 15179a8d927SPoul-Henning Kamp return (0); 15279a8d927SPoul-Henning Kamp } 15379a8d927SPoul-Henning Kamp 15479a8d927SPoul-Henning Kamp static void 15579a8d927SPoul-Henning Kamp uart_tty_close(struct tty *tp) 15679a8d927SPoul-Henning Kamp { 15779a8d927SPoul-Henning Kamp struct uart_softc *sc; 15879a8d927SPoul-Henning Kamp 159bc093719SEd Schouten sc = tty_softc(tp); 160fbbec42fSPoul-Henning Kamp if (sc == NULL || sc->sc_leaving || !sc->sc_opened) 16179a8d927SPoul-Henning Kamp return; 16279a8d927SPoul-Henning Kamp 16379a8d927SPoul-Henning Kamp if (sc->sc_hwiflow) 16479a8d927SPoul-Henning Kamp UART_IOCTL(sc, UART_IOCTL_IFLOW, 0); 16579a8d927SPoul-Henning Kamp if (sc->sc_hwoflow) 16679a8d927SPoul-Henning Kamp UART_IOCTL(sc, UART_IOCTL_OFLOW, 0); 16779a8d927SPoul-Henning Kamp if (sc->sc_sysdev == NULL) 16879a8d927SPoul-Henning Kamp UART_SETSIG(sc, SER_DDTR | SER_DRTS); 16979a8d927SPoul-Henning Kamp 17079a8d927SPoul-Henning Kamp wakeup(sc); 17179a8d927SPoul-Henning Kamp sc->sc_opened = 0; 17279a8d927SPoul-Henning Kamp return; 17379a8d927SPoul-Henning Kamp } 17479a8d927SPoul-Henning Kamp 17527d5dc18SMarcel Moolenaar static void 176bc093719SEd Schouten uart_tty_outwakeup(struct tty *tp) 17727d5dc18SMarcel Moolenaar { 17827d5dc18SMarcel Moolenaar struct uart_softc *sc; 17927d5dc18SMarcel Moolenaar 180bc093719SEd Schouten sc = tty_softc(tp); 18127d5dc18SMarcel Moolenaar if (sc == NULL || sc->sc_leaving) 18227d5dc18SMarcel Moolenaar return; 18327d5dc18SMarcel Moolenaar 184bc093719SEd Schouten if (sc->sc_txbusy) 18527d5dc18SMarcel Moolenaar return; 18627d5dc18SMarcel Moolenaar 187f1fb9647SMarcel Moolenaar /* 188f1fb9647SMarcel Moolenaar * Respect RTS/CTS (output) flow control if enabled and not already 189f1fb9647SMarcel Moolenaar * handled by hardware. 190f1fb9647SMarcel Moolenaar */ 191f1fb9647SMarcel Moolenaar if ((tp->t_termios.c_cflag & CCTS_OFLOW) && !sc->sc_hwoflow && 192f1fb9647SMarcel Moolenaar !(sc->sc_hwsig & SER_CTS)) 193f1fb9647SMarcel Moolenaar return; 194f1fb9647SMarcel Moolenaar 195bc093719SEd Schouten sc->sc_txdatasz = ttydisc_getc(tp, sc->sc_txbuf, sc->sc_txfifosz); 196bc093719SEd Schouten if (sc->sc_txdatasz != 0) 197bc093719SEd Schouten UART_TRANSMIT(sc); 19827d5dc18SMarcel Moolenaar } 19927d5dc18SMarcel Moolenaar 2000acb3c4aSMarcel Moolenaar static void 2010acb3c4aSMarcel Moolenaar uart_tty_inwakeup(struct tty *tp) 2020acb3c4aSMarcel Moolenaar { 2030acb3c4aSMarcel Moolenaar struct uart_softc *sc; 2040acb3c4aSMarcel Moolenaar 2050acb3c4aSMarcel Moolenaar sc = tty_softc(tp); 2060acb3c4aSMarcel Moolenaar if (sc == NULL || sc->sc_leaving) 2070acb3c4aSMarcel Moolenaar return; 2080acb3c4aSMarcel Moolenaar 2090acb3c4aSMarcel Moolenaar if (sc->sc_isquelch) { 2100acb3c4aSMarcel Moolenaar if ((tp->t_termios.c_cflag & CRTS_IFLOW) && !sc->sc_hwiflow) 2110acb3c4aSMarcel Moolenaar UART_SETSIG(sc, SER_DRTS|SER_RTS); 2120acb3c4aSMarcel Moolenaar sc->sc_isquelch = 0; 2130acb3c4aSMarcel Moolenaar uart_sched_softih(sc, SER_INT_RXREADY); 2140acb3c4aSMarcel Moolenaar } 2150acb3c4aSMarcel Moolenaar } 2160acb3c4aSMarcel Moolenaar 217bc093719SEd Schouten static int 218bc093719SEd Schouten uart_tty_ioctl(struct tty *tp, u_long cmd, caddr_t data, struct thread *td) 219bc093719SEd Schouten { 220bc093719SEd Schouten struct uart_softc *sc; 221bc093719SEd Schouten 222bc093719SEd Schouten sc = tty_softc(tp); 223bc093719SEd Schouten 224bc093719SEd Schouten switch (cmd) { 225bc093719SEd Schouten case TIOCSBRK: 226bc093719SEd Schouten UART_IOCTL(sc, UART_IOCTL_BREAK, 1); 227bc093719SEd Schouten return (0); 228bc093719SEd Schouten case TIOCCBRK: 229bc093719SEd Schouten UART_IOCTL(sc, UART_IOCTL_BREAK, 0); 230bc093719SEd Schouten return (0); 231bc093719SEd Schouten default: 232bc093719SEd Schouten return pps_ioctl(cmd, data, &sc->sc_pps); 233bc093719SEd Schouten } 23427d5dc18SMarcel Moolenaar } 23527d5dc18SMarcel Moolenaar 23627d5dc18SMarcel Moolenaar static int 23727d5dc18SMarcel Moolenaar uart_tty_param(struct tty *tp, struct termios *t) 23827d5dc18SMarcel Moolenaar { 23927d5dc18SMarcel Moolenaar struct uart_softc *sc; 24027d5dc18SMarcel Moolenaar int databits, parity, stopbits; 24127d5dc18SMarcel Moolenaar 242bc093719SEd Schouten sc = tty_softc(tp); 24327d5dc18SMarcel Moolenaar if (sc == NULL || sc->sc_leaving) 24427d5dc18SMarcel Moolenaar return (ENODEV); 24527d5dc18SMarcel Moolenaar if (t->c_ispeed != t->c_ospeed && t->c_ospeed != 0) 24627d5dc18SMarcel Moolenaar return (EINVAL); 24727d5dc18SMarcel Moolenaar if (t->c_ospeed == 0) { 24828710806SPoul-Henning Kamp UART_SETSIG(sc, SER_DDTR | SER_DRTS); 24927d5dc18SMarcel Moolenaar return (0); 25027d5dc18SMarcel Moolenaar } 25127d5dc18SMarcel Moolenaar switch (t->c_cflag & CSIZE) { 25227d5dc18SMarcel Moolenaar case CS5: databits = 5; break; 25327d5dc18SMarcel Moolenaar case CS6: databits = 6; break; 25427d5dc18SMarcel Moolenaar case CS7: databits = 7; break; 25527d5dc18SMarcel Moolenaar default: databits = 8; break; 25627d5dc18SMarcel Moolenaar } 25727d5dc18SMarcel Moolenaar stopbits = (t->c_cflag & CSTOPB) ? 2 : 1; 25827d5dc18SMarcel Moolenaar if (t->c_cflag & PARENB) 25927d5dc18SMarcel Moolenaar parity = (t->c_cflag & PARODD) ? UART_PARITY_ODD 26027d5dc18SMarcel Moolenaar : UART_PARITY_EVEN; 26127d5dc18SMarcel Moolenaar else 26227d5dc18SMarcel Moolenaar parity = UART_PARITY_NONE; 263b662bdc2SMarcel Moolenaar if (UART_PARAM(sc, t->c_ospeed, databits, stopbits, parity) != 0) 264b662bdc2SMarcel Moolenaar return (EINVAL); 26528710806SPoul-Henning Kamp UART_SETSIG(sc, SER_DDTR | SER_DTR); 26627d5dc18SMarcel Moolenaar /* Set input flow control state. */ 26727d5dc18SMarcel Moolenaar if (!sc->sc_hwiflow) { 2680acb3c4aSMarcel Moolenaar if ((t->c_cflag & CRTS_IFLOW) && sc->sc_isquelch) 26928710806SPoul-Henning Kamp UART_SETSIG(sc, SER_DRTS); 2700acb3c4aSMarcel Moolenaar else 27128710806SPoul-Henning Kamp UART_SETSIG(sc, SER_DRTS | SER_RTS); 27227d5dc18SMarcel Moolenaar } else 27327d5dc18SMarcel Moolenaar UART_IOCTL(sc, UART_IOCTL_IFLOW, (t->c_cflag & CRTS_IFLOW)); 27427d5dc18SMarcel Moolenaar /* Set output flow control state. */ 27527d5dc18SMarcel Moolenaar if (sc->sc_hwoflow) 27627d5dc18SMarcel Moolenaar UART_IOCTL(sc, UART_IOCTL_OFLOW, (t->c_cflag & CCTS_OFLOW)); 277bc093719SEd Schouten 27827d5dc18SMarcel Moolenaar return (0); 27927d5dc18SMarcel Moolenaar } 28027d5dc18SMarcel Moolenaar 28196df2b0bSPoul-Henning Kamp static int 28296df2b0bSPoul-Henning Kamp uart_tty_modem(struct tty *tp, int biton, int bitoff) 28396df2b0bSPoul-Henning Kamp { 28496df2b0bSPoul-Henning Kamp struct uart_softc *sc; 28596df2b0bSPoul-Henning Kamp 286bc093719SEd Schouten sc = tty_softc(tp); 28796df2b0bSPoul-Henning Kamp if (biton != 0 || bitoff != 0) 28896df2b0bSPoul-Henning Kamp UART_SETSIG(sc, SER_DELTA(bitoff|biton) | biton); 28996df2b0bSPoul-Henning Kamp return (sc->sc_hwsig); 29096df2b0bSPoul-Henning Kamp } 29196df2b0bSPoul-Henning Kamp 29227d5dc18SMarcel Moolenaar void 29327d5dc18SMarcel Moolenaar uart_tty_intr(void *arg) 29427d5dc18SMarcel Moolenaar { 29527d5dc18SMarcel Moolenaar struct uart_softc *sc = arg; 29627d5dc18SMarcel Moolenaar struct tty *tp; 297bc093719SEd Schouten int c, err = 0, pend, sig, xc; 29827d5dc18SMarcel Moolenaar 29927d5dc18SMarcel Moolenaar if (sc->sc_leaving) 30027d5dc18SMarcel Moolenaar return; 30127d5dc18SMarcel Moolenaar 30227d5dc18SMarcel Moolenaar pend = atomic_readandclear_32(&sc->sc_ttypend); 3032d511805SMarcel Moolenaar if (!(pend & SER_INT_MASK)) 30427d5dc18SMarcel Moolenaar return; 30527d5dc18SMarcel Moolenaar 30627d5dc18SMarcel Moolenaar tp = sc->sc_u.u_tty.tp; 307bc093719SEd Schouten tty_lock(tp); 30827d5dc18SMarcel Moolenaar 3092d511805SMarcel Moolenaar if (pend & SER_INT_RXREADY) { 3100acb3c4aSMarcel Moolenaar while (!uart_rx_empty(sc) && !sc->sc_isquelch) { 3110acb3c4aSMarcel Moolenaar xc = uart_rx_peek(sc); 31227d5dc18SMarcel Moolenaar c = xc & 0xff; 31327d5dc18SMarcel Moolenaar if (xc & UART_STAT_FRAMERR) 314bc093719SEd Schouten err |= TRE_FRAMING; 315dd5b096fSMarcel Moolenaar if (xc & UART_STAT_OVERRUN) 316bc093719SEd Schouten err |= TRE_OVERRUN; 31727d5dc18SMarcel Moolenaar if (xc & UART_STAT_PARERR) 318bc093719SEd Schouten err |= TRE_PARITY; 3190acb3c4aSMarcel Moolenaar if (ttydisc_rint(tp, c, err) != 0) { 3200acb3c4aSMarcel Moolenaar sc->sc_isquelch = 1; 3210acb3c4aSMarcel Moolenaar if ((tp->t_termios.c_cflag & CRTS_IFLOW) && 3220acb3c4aSMarcel Moolenaar !sc->sc_hwiflow) 3230acb3c4aSMarcel Moolenaar UART_SETSIG(sc, SER_DRTS); 3240acb3c4aSMarcel Moolenaar } else 3250acb3c4aSMarcel Moolenaar uart_rx_next(sc); 32627d5dc18SMarcel Moolenaar } 32727d5dc18SMarcel Moolenaar } 32827d5dc18SMarcel Moolenaar 329bc093719SEd Schouten if (pend & SER_INT_BREAK) 330bc093719SEd Schouten ttydisc_rint(tp, 0, TRE_BREAK); 33127d5dc18SMarcel Moolenaar 3322d511805SMarcel Moolenaar if (pend & SER_INT_SIGCHG) { 3332d511805SMarcel Moolenaar sig = pend & SER_INT_SIGMASK; 33428710806SPoul-Henning Kamp if (sig & SER_DDCD) 335bc093719SEd Schouten ttydisc_modem(tp, sig & SER_DCD); 336f1fb9647SMarcel Moolenaar if (sig & SER_DCTS) 337bc093719SEd Schouten uart_tty_outwakeup(tp); 33827d5dc18SMarcel Moolenaar } 33927d5dc18SMarcel Moolenaar 340bc093719SEd Schouten if (pend & SER_INT_TXIDLE) 341bc093719SEd Schouten uart_tty_outwakeup(tp); 342bc093719SEd Schouten ttydisc_rint_done(tp); 343bc093719SEd Schouten tty_unlock(tp); 34427d5dc18SMarcel Moolenaar } 345bc093719SEd Schouten 3469b866e4eSEd Schouten static void 3479b866e4eSEd Schouten uart_tty_free(void *arg) 3489b866e4eSEd Schouten { 3499b866e4eSEd Schouten 3509b866e4eSEd Schouten /* 3519b866e4eSEd Schouten * XXX: uart(4) could reuse the device unit number before it is 3529b866e4eSEd Schouten * being freed by the TTY layer. We should use this hook to free 3539b866e4eSEd Schouten * the device unit number, but unfortunately newbus does not 3549b866e4eSEd Schouten * seem to support such a construct. 3559b866e4eSEd Schouten */ 3569b866e4eSEd Schouten } 3579b866e4eSEd Schouten 358bc093719SEd Schouten static struct ttydevsw uart_tty_class = { 359bc093719SEd Schouten .tsw_flags = TF_INITLOCK|TF_CALLOUT, 360bc093719SEd Schouten .tsw_open = uart_tty_open, 361bc093719SEd Schouten .tsw_close = uart_tty_close, 362bc093719SEd Schouten .tsw_outwakeup = uart_tty_outwakeup, 3630acb3c4aSMarcel Moolenaar .tsw_inwakeup = uart_tty_inwakeup, 364bc093719SEd Schouten .tsw_ioctl = uart_tty_ioctl, 365bc093719SEd Schouten .tsw_param = uart_tty_param, 366bc093719SEd Schouten .tsw_modem = uart_tty_modem, 3679b866e4eSEd Schouten .tsw_free = uart_tty_free, 368bc093719SEd Schouten }; 36927d5dc18SMarcel Moolenaar 37027d5dc18SMarcel Moolenaar int 37127d5dc18SMarcel Moolenaar uart_tty_attach(struct uart_softc *sc) 37227d5dc18SMarcel Moolenaar { 37327d5dc18SMarcel Moolenaar struct tty *tp; 37479a8d927SPoul-Henning Kamp int unit; 37527d5dc18SMarcel Moolenaar 376c5e30cc0SEd Schouten sc->sc_u.u_tty.tp = tp = tty_alloc(&uart_tty_class, sc); 37727d5dc18SMarcel Moolenaar 37879a8d927SPoul-Henning Kamp unit = device_get_unit(sc->sc_dev); 37927d5dc18SMarcel Moolenaar 38027d5dc18SMarcel Moolenaar if (sc->sc_sysdev != NULL && sc->sc_sysdev->type == UART_DEV_CONSOLE) { 38112d984b6SMarcel Moolenaar sprintf(((struct consdev *)sc->sc_sysdev->cookie)->cn_name, 38279a8d927SPoul-Henning Kamp "ttyu%r", unit); 383f725b213SMarcel Moolenaar tty_init_console(tp, sc->sc_sysdev->baudrate); 38427d5dc18SMarcel Moolenaar } 38527d5dc18SMarcel Moolenaar 386e0f66ef8SJohn Baldwin swi_add(&tty_intr_event, uart_driver_name, uart_tty_intr, sc, SWI_TTY, 38727d5dc18SMarcel Moolenaar INTR_TYPE_TTY, &sc->sc_softih); 38827d5dc18SMarcel Moolenaar 389bc093719SEd Schouten tty_makedev(tp, NULL, "u%r", unit); 39079a8d927SPoul-Henning Kamp 39127d5dc18SMarcel Moolenaar return (0); 39227d5dc18SMarcel Moolenaar } 39327d5dc18SMarcel Moolenaar 3947d376cbcSAlexander Kabaev int 3957d376cbcSAlexander Kabaev uart_tty_detach(struct uart_softc *sc) 39627d5dc18SMarcel Moolenaar { 39779a8d927SPoul-Henning Kamp struct tty *tp; 39827d5dc18SMarcel Moolenaar 39979a8d927SPoul-Henning Kamp tp = sc->sc_u.u_tty.tp; 400bc093719SEd Schouten 401bc093719SEd Schouten tty_lock(tp); 402284b6708SJohn Baldwin swi_remove(sc->sc_softih); 403bc093719SEd Schouten tty_rel_gone(tp); 40427d5dc18SMarcel Moolenaar 40527d5dc18SMarcel Moolenaar return (0); 40627d5dc18SMarcel Moolenaar } 407*b59236ceSIan Lepore 408*b59236ceSIan Lepore struct mtx * 409*b59236ceSIan Lepore uart_tty_getlock(struct uart_softc *sc) 410*b59236ceSIan Lepore { 411*b59236ceSIan Lepore 412*b59236ceSIan Lepore if (sc->sc_u.u_tty.tp != NULL) 413*b59236ceSIan Lepore return (tty_getlock(sc->sc_u.u_tty.tp)); 414*b59236ceSIan Lepore else 415*b59236ceSIan Lepore return (NULL); 416*b59236ceSIan Lepore } 417