1098ca2bdSWarner Losh /*-
24d846d26SWarner Losh * SPDX-License-Identifier: BSD-2-Clause
3718cf2ccSPedro F. Giffuni *
427d5dc18SMarcel Moolenaar * Copyright (c) 2003 Marcel Moolenaar
527d5dc18SMarcel Moolenaar * All rights reserved.
627d5dc18SMarcel Moolenaar *
727d5dc18SMarcel Moolenaar * Redistribution and use in source and binary forms, with or without
827d5dc18SMarcel Moolenaar * modification, are permitted provided that the following conditions
927d5dc18SMarcel Moolenaar * are met:
1027d5dc18SMarcel Moolenaar *
1127d5dc18SMarcel Moolenaar * 1. Redistributions of source code must retain the above copyright
1227d5dc18SMarcel Moolenaar * notice, this list of conditions and the following disclaimer.
1327d5dc18SMarcel Moolenaar * 2. Redistributions in binary form must reproduce the above copyright
1427d5dc18SMarcel Moolenaar * notice, this list of conditions and the following disclaimer in the
1527d5dc18SMarcel Moolenaar * documentation and/or other materials provided with the distribution.
1627d5dc18SMarcel Moolenaar *
1727d5dc18SMarcel Moolenaar * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
1827d5dc18SMarcel Moolenaar * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
1927d5dc18SMarcel Moolenaar * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
2027d5dc18SMarcel Moolenaar * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
2127d5dc18SMarcel Moolenaar * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
2227d5dc18SMarcel Moolenaar * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
2327d5dc18SMarcel Moolenaar * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
2427d5dc18SMarcel Moolenaar * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
2527d5dc18SMarcel Moolenaar * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
2627d5dc18SMarcel Moolenaar * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
2727d5dc18SMarcel Moolenaar */
2827d5dc18SMarcel Moolenaar
2927d5dc18SMarcel Moolenaar #include <sys/cdefs.h>
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 <machine/bus.h>
3527d5dc18SMarcel Moolenaar
3627d5dc18SMarcel Moolenaar #include <dev/uart/uart.h>
3727d5dc18SMarcel Moolenaar #include <dev/uart/uart_cpu.h>
3827d5dc18SMarcel Moolenaar #include <dev/uart/uart_bus.h>
390204ed7aSMarcel Moolenaar
400204ed7aSMarcel Moolenaar #include <dev/ic/z8530.h>
4127d5dc18SMarcel Moolenaar
4227d5dc18SMarcel Moolenaar #include "uart_if.h"
4327d5dc18SMarcel Moolenaar
4427d5dc18SMarcel Moolenaar #define DEFAULT_RCLK 307200
4527d5dc18SMarcel Moolenaar
4654cfafcfSMarcel Moolenaar /* Hack! */
4754cfafcfSMarcel Moolenaar #ifdef __powerpc__
4854cfafcfSMarcel Moolenaar #define UART_PCLK 0
4954cfafcfSMarcel Moolenaar #else
5054cfafcfSMarcel Moolenaar #define UART_PCLK MCB2_PCLK
5154cfafcfSMarcel Moolenaar #endif
5254cfafcfSMarcel Moolenaar
5327d5dc18SMarcel Moolenaar /* Multiplexed I/O. */
5427d5dc18SMarcel Moolenaar static __inline void
uart_setmreg(struct uart_bas * bas,int reg,int val)5527d5dc18SMarcel Moolenaar uart_setmreg(struct uart_bas *bas, int reg, int val)
5627d5dc18SMarcel Moolenaar {
5727d5dc18SMarcel Moolenaar
5827d5dc18SMarcel Moolenaar uart_setreg(bas, REG_CTRL, reg);
5927d5dc18SMarcel Moolenaar uart_barrier(bas);
6027d5dc18SMarcel Moolenaar uart_setreg(bas, REG_CTRL, val);
6127d5dc18SMarcel Moolenaar }
6227d5dc18SMarcel Moolenaar
6327d5dc18SMarcel Moolenaar static __inline uint8_t
uart_getmreg(struct uart_bas * bas,int reg)6427d5dc18SMarcel Moolenaar uart_getmreg(struct uart_bas *bas, int reg)
6527d5dc18SMarcel Moolenaar {
6627d5dc18SMarcel Moolenaar
6727d5dc18SMarcel Moolenaar uart_setreg(bas, REG_CTRL, reg);
6827d5dc18SMarcel Moolenaar uart_barrier(bas);
6927d5dc18SMarcel Moolenaar return (uart_getreg(bas, REG_CTRL));
7027d5dc18SMarcel Moolenaar }
7127d5dc18SMarcel Moolenaar
7227d5dc18SMarcel Moolenaar static int
z8530_divisor(int rclk,int baudrate)7327d5dc18SMarcel Moolenaar z8530_divisor(int rclk, int baudrate)
7427d5dc18SMarcel Moolenaar {
7527d5dc18SMarcel Moolenaar int act_baud, divisor, error;
7627d5dc18SMarcel Moolenaar
7727d5dc18SMarcel Moolenaar if (baudrate == 0)
78ace86f3fSMarcel Moolenaar return (-1);
7927d5dc18SMarcel Moolenaar
8027d5dc18SMarcel Moolenaar divisor = (rclk + baudrate) / (baudrate << 1) - 2;
81a24d2e12SMarcel Moolenaar if (divisor < 0 || divisor >= 65536)
82ace86f3fSMarcel Moolenaar return (-1);
8327d5dc18SMarcel Moolenaar act_baud = rclk / 2 / (divisor + 2);
8427d5dc18SMarcel Moolenaar
8527d5dc18SMarcel Moolenaar /* 10 times error in percent: */
8627d5dc18SMarcel Moolenaar error = ((act_baud - baudrate) * 2000 / baudrate + 1) >> 1;
8727d5dc18SMarcel Moolenaar
8827d5dc18SMarcel Moolenaar /* 3.0% maximum error tolerance: */
8927d5dc18SMarcel Moolenaar if (error < -30 || error > 30)
90ace86f3fSMarcel Moolenaar return (-1);
9127d5dc18SMarcel Moolenaar
9227d5dc18SMarcel Moolenaar return (divisor);
9327d5dc18SMarcel Moolenaar }
9427d5dc18SMarcel Moolenaar
9527d5dc18SMarcel Moolenaar static int
z8530_param(struct uart_bas * bas,int baudrate,int databits,int stopbits,int parity,uint8_t * tpcp)9627d5dc18SMarcel Moolenaar z8530_param(struct uart_bas *bas, int baudrate, int databits, int stopbits,
9727d5dc18SMarcel Moolenaar int parity, uint8_t *tpcp)
9827d5dc18SMarcel Moolenaar {
9927d5dc18SMarcel Moolenaar int divisor;
10027d5dc18SMarcel Moolenaar uint8_t mpm, rpc, tpc;
10127d5dc18SMarcel Moolenaar
10227d5dc18SMarcel Moolenaar rpc = RPC_RXE;
10327d5dc18SMarcel Moolenaar mpm = MPM_CM16;
10427d5dc18SMarcel Moolenaar tpc = TPC_TXE | (*tpcp & (TPC_DTR | TPC_RTS));
10527d5dc18SMarcel Moolenaar
10627d5dc18SMarcel Moolenaar if (databits >= 8) {
10727d5dc18SMarcel Moolenaar rpc |= RPC_RB8;
10827d5dc18SMarcel Moolenaar tpc |= TPC_TB8;
10927d5dc18SMarcel Moolenaar } else if (databits == 7) {
11027d5dc18SMarcel Moolenaar rpc |= RPC_RB7;
11127d5dc18SMarcel Moolenaar tpc |= TPC_TB7;
11227d5dc18SMarcel Moolenaar } else if (databits == 6) {
11327d5dc18SMarcel Moolenaar rpc |= RPC_RB6;
11427d5dc18SMarcel Moolenaar tpc |= TPC_TB6;
11527d5dc18SMarcel Moolenaar } else {
11627d5dc18SMarcel Moolenaar rpc |= RPC_RB5;
11727d5dc18SMarcel Moolenaar tpc |= TPC_TB5;
11827d5dc18SMarcel Moolenaar }
11927d5dc18SMarcel Moolenaar mpm |= (stopbits > 1) ? MPM_SB2 : MPM_SB1;
12027d5dc18SMarcel Moolenaar switch (parity) {
12127d5dc18SMarcel Moolenaar case UART_PARITY_EVEN: mpm |= MPM_PE | MPM_EVEN; break;
12227d5dc18SMarcel Moolenaar case UART_PARITY_NONE: break;
12327d5dc18SMarcel Moolenaar case UART_PARITY_ODD: mpm |= MPM_PE; break;
12427d5dc18SMarcel Moolenaar default: return (EINVAL);
12527d5dc18SMarcel Moolenaar }
12627d5dc18SMarcel Moolenaar
12727d5dc18SMarcel Moolenaar if (baudrate > 0) {
12827d5dc18SMarcel Moolenaar divisor = z8530_divisor(bas->rclk, baudrate);
129ace86f3fSMarcel Moolenaar if (divisor == -1)
13027d5dc18SMarcel Moolenaar return (EINVAL);
131ace86f3fSMarcel Moolenaar } else
132ace86f3fSMarcel Moolenaar divisor = -1;
133ace86f3fSMarcel Moolenaar
13454cfafcfSMarcel Moolenaar uart_setmreg(bas, WR_MCB2, UART_PCLK);
135ace86f3fSMarcel Moolenaar uart_barrier(bas);
136ace86f3fSMarcel Moolenaar
137ace86f3fSMarcel Moolenaar if (divisor >= 0) {
13827d5dc18SMarcel Moolenaar uart_setmreg(bas, WR_TCL, divisor & 0xff);
13927d5dc18SMarcel Moolenaar uart_barrier(bas);
14027d5dc18SMarcel Moolenaar uart_setmreg(bas, WR_TCH, (divisor >> 8) & 0xff);
14127d5dc18SMarcel Moolenaar uart_barrier(bas);
14227d5dc18SMarcel Moolenaar }
14327d5dc18SMarcel Moolenaar
14427d5dc18SMarcel Moolenaar uart_setmreg(bas, WR_RPC, rpc);
14527d5dc18SMarcel Moolenaar uart_barrier(bas);
14627d5dc18SMarcel Moolenaar uart_setmreg(bas, WR_MPM, mpm);
14727d5dc18SMarcel Moolenaar uart_barrier(bas);
14827d5dc18SMarcel Moolenaar uart_setmreg(bas, WR_TPC, tpc);
14927d5dc18SMarcel Moolenaar uart_barrier(bas);
15054cfafcfSMarcel Moolenaar uart_setmreg(bas, WR_MCB2, UART_PCLK | MCB2_BRGE);
151ace86f3fSMarcel Moolenaar uart_barrier(bas);
15227d5dc18SMarcel Moolenaar *tpcp = tpc;
15327d5dc18SMarcel Moolenaar return (0);
15427d5dc18SMarcel Moolenaar }
15527d5dc18SMarcel Moolenaar
15627d5dc18SMarcel Moolenaar static int
z8530_setup(struct uart_bas * bas,int baudrate,int databits,int stopbits,int parity)15727d5dc18SMarcel Moolenaar z8530_setup(struct uart_bas *bas, int baudrate, int databits, int stopbits,
15827d5dc18SMarcel Moolenaar int parity)
15927d5dc18SMarcel Moolenaar {
160c3c16fcbSMarcel Moolenaar uint8_t tpc;
16127d5dc18SMarcel Moolenaar
16227d5dc18SMarcel Moolenaar if (bas->rclk == 0)
16327d5dc18SMarcel Moolenaar bas->rclk = DEFAULT_RCLK;
16427d5dc18SMarcel Moolenaar
16527d5dc18SMarcel Moolenaar /* Assume we don't need to perform a full hardware reset. */
166875f70dbSMarcel Moolenaar switch (bas->chan) {
167875f70dbSMarcel Moolenaar case 1:
168c3c16fcbSMarcel Moolenaar uart_setmreg(bas, WR_MIC, MIC_NV | MIC_CRA);
169875f70dbSMarcel Moolenaar break;
170875f70dbSMarcel Moolenaar case 2:
171c3c16fcbSMarcel Moolenaar uart_setmreg(bas, WR_MIC, MIC_NV | MIC_CRB);
172875f70dbSMarcel Moolenaar break;
173875f70dbSMarcel Moolenaar }
17427d5dc18SMarcel Moolenaar uart_barrier(bas);
175ace86f3fSMarcel Moolenaar /* Set clock sources. */
17627d5dc18SMarcel Moolenaar uart_setmreg(bas, WR_CMC, CMC_RC_BRG | CMC_TC_BRG);
17754cfafcfSMarcel Moolenaar uart_setmreg(bas, WR_MCB2, UART_PCLK);
17827d5dc18SMarcel Moolenaar uart_barrier(bas);
17927d5dc18SMarcel Moolenaar /* Set data encoding. */
18027d5dc18SMarcel Moolenaar uart_setmreg(bas, WR_MCB1, MCB1_NRZ);
18127d5dc18SMarcel Moolenaar uart_barrier(bas);
18227d5dc18SMarcel Moolenaar
18327d5dc18SMarcel Moolenaar tpc = TPC_DTR | TPC_RTS;
18427d5dc18SMarcel Moolenaar z8530_param(bas, baudrate, databits, stopbits, parity, &tpc);
18527d5dc18SMarcel Moolenaar return (int)tpc;
18627d5dc18SMarcel Moolenaar }
18727d5dc18SMarcel Moolenaar
18827d5dc18SMarcel Moolenaar /*
18927d5dc18SMarcel Moolenaar * Low-level UART interface.
19027d5dc18SMarcel Moolenaar */
19127d5dc18SMarcel Moolenaar static int z8530_probe(struct uart_bas *bas);
19227d5dc18SMarcel Moolenaar static void z8530_init(struct uart_bas *bas, int, int, int, int);
19327d5dc18SMarcel Moolenaar static void z8530_term(struct uart_bas *bas);
19427d5dc18SMarcel Moolenaar static void z8530_putc(struct uart_bas *bas, int);
19597202af2SMarius Strobl static int z8530_rxready(struct uart_bas *bas);
196634e63c9SMarcel Moolenaar static int z8530_getc(struct uart_bas *bas, struct mtx *);
19727d5dc18SMarcel Moolenaar
198f8100ce2SMarcel Moolenaar static struct uart_ops uart_z8530_ops = {
19927d5dc18SMarcel Moolenaar .probe = z8530_probe,
20027d5dc18SMarcel Moolenaar .init = z8530_init,
20127d5dc18SMarcel Moolenaar .term = z8530_term,
20227d5dc18SMarcel Moolenaar .putc = z8530_putc,
20397202af2SMarius Strobl .rxready = z8530_rxready,
20427d5dc18SMarcel Moolenaar .getc = z8530_getc,
20527d5dc18SMarcel Moolenaar };
20627d5dc18SMarcel Moolenaar
20727d5dc18SMarcel Moolenaar static int
z8530_probe(struct uart_bas * bas)20827d5dc18SMarcel Moolenaar z8530_probe(struct uart_bas *bas)
20927d5dc18SMarcel Moolenaar {
21027d5dc18SMarcel Moolenaar
21127d5dc18SMarcel Moolenaar return (0);
21227d5dc18SMarcel Moolenaar }
21327d5dc18SMarcel Moolenaar
21427d5dc18SMarcel Moolenaar static void
z8530_init(struct uart_bas * bas,int baudrate,int databits,int stopbits,int parity)21527d5dc18SMarcel Moolenaar z8530_init(struct uart_bas *bas, int baudrate, int databits, int stopbits,
21627d5dc18SMarcel Moolenaar int parity)
21727d5dc18SMarcel Moolenaar {
21827d5dc18SMarcel Moolenaar
21927d5dc18SMarcel Moolenaar z8530_setup(bas, baudrate, databits, stopbits, parity);
22027d5dc18SMarcel Moolenaar }
22127d5dc18SMarcel Moolenaar
22227d5dc18SMarcel Moolenaar static void
z8530_term(struct uart_bas * bas)22327d5dc18SMarcel Moolenaar z8530_term(struct uart_bas *bas)
22427d5dc18SMarcel Moolenaar {
22527d5dc18SMarcel Moolenaar }
22627d5dc18SMarcel Moolenaar
22727d5dc18SMarcel Moolenaar static void
z8530_putc(struct uart_bas * bas,int c)22827d5dc18SMarcel Moolenaar z8530_putc(struct uart_bas *bas, int c)
22927d5dc18SMarcel Moolenaar {
23027d5dc18SMarcel Moolenaar
231ca83142fSMarcel Moolenaar while (!(uart_getreg(bas, REG_CTRL) & BES_TXE))
23227d5dc18SMarcel Moolenaar ;
23327d5dc18SMarcel Moolenaar uart_setreg(bas, REG_DATA, c);
23427d5dc18SMarcel Moolenaar uart_barrier(bas);
23527d5dc18SMarcel Moolenaar }
23627d5dc18SMarcel Moolenaar
23727d5dc18SMarcel Moolenaar static int
z8530_rxready(struct uart_bas * bas)23897202af2SMarius Strobl z8530_rxready(struct uart_bas *bas)
23927d5dc18SMarcel Moolenaar {
24027d5dc18SMarcel Moolenaar
24197202af2SMarius Strobl return ((uart_getreg(bas, REG_CTRL) & BES_RXA) != 0 ? 1 : 0);
24227d5dc18SMarcel Moolenaar }
24327d5dc18SMarcel Moolenaar
24427d5dc18SMarcel Moolenaar static int
z8530_getc(struct uart_bas * bas,struct mtx * hwmtx)245634e63c9SMarcel Moolenaar z8530_getc(struct uart_bas *bas, struct mtx *hwmtx)
24627d5dc18SMarcel Moolenaar {
247634e63c9SMarcel Moolenaar int c;
24827d5dc18SMarcel Moolenaar
249634e63c9SMarcel Moolenaar uart_lock(hwmtx);
250634e63c9SMarcel Moolenaar
251634e63c9SMarcel Moolenaar while (!(uart_getreg(bas, REG_CTRL) & BES_RXA)) {
252634e63c9SMarcel Moolenaar uart_unlock(hwmtx);
253634e63c9SMarcel Moolenaar DELAY(10);
254634e63c9SMarcel Moolenaar uart_lock(hwmtx);
255634e63c9SMarcel Moolenaar }
256634e63c9SMarcel Moolenaar
257634e63c9SMarcel Moolenaar c = uart_getreg(bas, REG_DATA);
258634e63c9SMarcel Moolenaar
259634e63c9SMarcel Moolenaar uart_unlock(hwmtx);
260634e63c9SMarcel Moolenaar
261634e63c9SMarcel Moolenaar return (c);
26227d5dc18SMarcel Moolenaar }
26327d5dc18SMarcel Moolenaar
26427d5dc18SMarcel Moolenaar /*
26527d5dc18SMarcel Moolenaar * High-level UART interface.
26627d5dc18SMarcel Moolenaar */
26727d5dc18SMarcel Moolenaar struct z8530_softc {
26827d5dc18SMarcel Moolenaar struct uart_softc base;
26927d5dc18SMarcel Moolenaar uint8_t tpc;
270eced4286SMarcel Moolenaar uint8_t txidle;
27127d5dc18SMarcel Moolenaar };
27227d5dc18SMarcel Moolenaar
27327d5dc18SMarcel Moolenaar static int z8530_bus_attach(struct uart_softc *);
27427d5dc18SMarcel Moolenaar static int z8530_bus_detach(struct uart_softc *);
27527d5dc18SMarcel Moolenaar static int z8530_bus_flush(struct uart_softc *, int);
27627d5dc18SMarcel Moolenaar static int z8530_bus_getsig(struct uart_softc *);
27727d5dc18SMarcel Moolenaar static int z8530_bus_ioctl(struct uart_softc *, int, intptr_t);
27827d5dc18SMarcel Moolenaar static int z8530_bus_ipend(struct uart_softc *);
27927d5dc18SMarcel Moolenaar static int z8530_bus_param(struct uart_softc *, int, int, int, int);
28027d5dc18SMarcel Moolenaar static int z8530_bus_probe(struct uart_softc *);
28127d5dc18SMarcel Moolenaar static int z8530_bus_receive(struct uart_softc *);
28227d5dc18SMarcel Moolenaar static int z8530_bus_setsig(struct uart_softc *, int);
28327d5dc18SMarcel Moolenaar static int z8530_bus_transmit(struct uart_softc *);
284d76a1ef4SWarner Losh static void z8530_bus_grab(struct uart_softc *);
285d76a1ef4SWarner Losh static void z8530_bus_ungrab(struct uart_softc *);
28627d5dc18SMarcel Moolenaar
28727d5dc18SMarcel Moolenaar static kobj_method_t z8530_methods[] = {
28827d5dc18SMarcel Moolenaar KOBJMETHOD(uart_attach, z8530_bus_attach),
28927d5dc18SMarcel Moolenaar KOBJMETHOD(uart_detach, z8530_bus_detach),
29027d5dc18SMarcel Moolenaar KOBJMETHOD(uart_flush, z8530_bus_flush),
29127d5dc18SMarcel Moolenaar KOBJMETHOD(uart_getsig, z8530_bus_getsig),
29227d5dc18SMarcel Moolenaar KOBJMETHOD(uart_ioctl, z8530_bus_ioctl),
29327d5dc18SMarcel Moolenaar KOBJMETHOD(uart_ipend, z8530_bus_ipend),
29427d5dc18SMarcel Moolenaar KOBJMETHOD(uart_param, z8530_bus_param),
29527d5dc18SMarcel Moolenaar KOBJMETHOD(uart_probe, z8530_bus_probe),
29627d5dc18SMarcel Moolenaar KOBJMETHOD(uart_receive, z8530_bus_receive),
29727d5dc18SMarcel Moolenaar KOBJMETHOD(uart_setsig, z8530_bus_setsig),
29827d5dc18SMarcel Moolenaar KOBJMETHOD(uart_transmit, z8530_bus_transmit),
299d76a1ef4SWarner Losh KOBJMETHOD(uart_grab, z8530_bus_grab),
300d76a1ef4SWarner Losh KOBJMETHOD(uart_ungrab, z8530_bus_ungrab),
30127d5dc18SMarcel Moolenaar { 0, 0 }
30227d5dc18SMarcel Moolenaar };
30327d5dc18SMarcel Moolenaar
30427d5dc18SMarcel Moolenaar struct uart_class uart_z8530_class = {
305f8100ce2SMarcel Moolenaar "z8530",
30627d5dc18SMarcel Moolenaar z8530_methods,
30727d5dc18SMarcel Moolenaar sizeof(struct z8530_softc),
308f8100ce2SMarcel Moolenaar .uc_ops = &uart_z8530_ops,
30927d5dc18SMarcel Moolenaar .uc_range = 2,
310405ada37SAndrew Turner .uc_rclk = DEFAULT_RCLK,
311405ada37SAndrew Turner .uc_rshift = 0
31227d5dc18SMarcel Moolenaar };
31327d5dc18SMarcel Moolenaar
31427d5dc18SMarcel Moolenaar #define SIGCHG(c, i, s, d) \
31527d5dc18SMarcel Moolenaar if (c) { \
31627d5dc18SMarcel Moolenaar i |= (i & s) ? s : s | d; \
31727d5dc18SMarcel Moolenaar } else { \
31827d5dc18SMarcel Moolenaar i = (i & s) ? (i & ~s) | d : i; \
31927d5dc18SMarcel Moolenaar }
32027d5dc18SMarcel Moolenaar
32127d5dc18SMarcel Moolenaar static int
z8530_bus_attach(struct uart_softc * sc)32227d5dc18SMarcel Moolenaar z8530_bus_attach(struct uart_softc *sc)
32327d5dc18SMarcel Moolenaar {
32427d5dc18SMarcel Moolenaar struct z8530_softc *z8530 = (struct z8530_softc*)sc;
32527d5dc18SMarcel Moolenaar struct uart_bas *bas;
32627d5dc18SMarcel Moolenaar struct uart_devinfo *di;
32727d5dc18SMarcel Moolenaar
32827d5dc18SMarcel Moolenaar bas = &sc->sc_bas;
32927d5dc18SMarcel Moolenaar if (sc->sc_sysdev != NULL) {
33027d5dc18SMarcel Moolenaar di = sc->sc_sysdev;
33127d5dc18SMarcel Moolenaar z8530->tpc = TPC_DTR|TPC_RTS;
33227d5dc18SMarcel Moolenaar z8530_param(bas, di->baudrate, di->databits, di->stopbits,
33327d5dc18SMarcel Moolenaar di->parity, &z8530->tpc);
33427d5dc18SMarcel Moolenaar } else {
33527d5dc18SMarcel Moolenaar z8530->tpc = z8530_setup(bas, 9600, 8, 1, UART_PARITY_NONE);
33627d5dc18SMarcel Moolenaar z8530->tpc &= ~(TPC_DTR|TPC_RTS);
33727d5dc18SMarcel Moolenaar }
3382d511805SMarcel Moolenaar z8530->txidle = 1; /* Report SER_INT_TXIDLE. */
33927d5dc18SMarcel Moolenaar
34027d5dc18SMarcel Moolenaar (void)z8530_bus_getsig(sc);
34127d5dc18SMarcel Moolenaar
34227d5dc18SMarcel Moolenaar uart_setmreg(bas, WR_IC, IC_BRK | IC_CTS | IC_DCD);
34327d5dc18SMarcel Moolenaar uart_barrier(bas);
344c3c16fcbSMarcel Moolenaar uart_setmreg(bas, WR_IDT, IDT_XIE | IDT_TIE | IDT_RIA);
34527d5dc18SMarcel Moolenaar uart_barrier(bas);
34627d5dc18SMarcel Moolenaar uart_setmreg(bas, WR_IV, 0);
34727d5dc18SMarcel Moolenaar uart_barrier(bas);
34827d5dc18SMarcel Moolenaar uart_setmreg(bas, WR_TPC, z8530->tpc);
34927d5dc18SMarcel Moolenaar uart_barrier(bas);
350c3c16fcbSMarcel Moolenaar uart_setmreg(bas, WR_MIC, MIC_NV | MIC_MIE);
351c3c16fcbSMarcel Moolenaar uart_barrier(bas);
35227d5dc18SMarcel Moolenaar return (0);
35327d5dc18SMarcel Moolenaar }
35427d5dc18SMarcel Moolenaar
35527d5dc18SMarcel Moolenaar static int
z8530_bus_detach(struct uart_softc * sc)35627d5dc18SMarcel Moolenaar z8530_bus_detach(struct uart_softc *sc)
35727d5dc18SMarcel Moolenaar {
35827d5dc18SMarcel Moolenaar
35927d5dc18SMarcel Moolenaar return (0);
36027d5dc18SMarcel Moolenaar }
36127d5dc18SMarcel Moolenaar
36227d5dc18SMarcel Moolenaar static int
z8530_bus_flush(struct uart_softc * sc,int what)36327d5dc18SMarcel Moolenaar z8530_bus_flush(struct uart_softc *sc, int what)
36427d5dc18SMarcel Moolenaar {
36527d5dc18SMarcel Moolenaar
36627d5dc18SMarcel Moolenaar return (0);
36727d5dc18SMarcel Moolenaar }
36827d5dc18SMarcel Moolenaar
36927d5dc18SMarcel Moolenaar static int
z8530_bus_getsig(struct uart_softc * sc)37027d5dc18SMarcel Moolenaar z8530_bus_getsig(struct uart_softc *sc)
37127d5dc18SMarcel Moolenaar {
37227d5dc18SMarcel Moolenaar uint32_t new, old, sig;
37327d5dc18SMarcel Moolenaar uint8_t bes;
37427d5dc18SMarcel Moolenaar
37527d5dc18SMarcel Moolenaar do {
37627d5dc18SMarcel Moolenaar old = sc->sc_hwsig;
37727d5dc18SMarcel Moolenaar sig = old;
3788af03381SMarcel Moolenaar uart_lock(sc->sc_hwmtx);
37906287620SMarcel Moolenaar bes = uart_getmreg(&sc->sc_bas, RR_BES);
3808af03381SMarcel Moolenaar uart_unlock(sc->sc_hwmtx);
38128710806SPoul-Henning Kamp SIGCHG(bes & BES_CTS, sig, SER_CTS, SER_DCTS);
38228710806SPoul-Henning Kamp SIGCHG(bes & BES_DCD, sig, SER_DCD, SER_DDCD);
383c3c16fcbSMarcel Moolenaar SIGCHG(bes & BES_SYNC, sig, SER_DSR, SER_DDSR);
384ea549414SMarcel Moolenaar new = sig & ~SER_MASK_DELTA;
38527d5dc18SMarcel Moolenaar } while (!atomic_cmpset_32(&sc->sc_hwsig, old, new));
38627d5dc18SMarcel Moolenaar return (sig);
38727d5dc18SMarcel Moolenaar }
38827d5dc18SMarcel Moolenaar
38927d5dc18SMarcel Moolenaar static int
z8530_bus_ioctl(struct uart_softc * sc,int request,intptr_t data)39027d5dc18SMarcel Moolenaar z8530_bus_ioctl(struct uart_softc *sc, int request, intptr_t data)
39127d5dc18SMarcel Moolenaar {
39227d5dc18SMarcel Moolenaar struct z8530_softc *z8530 = (struct z8530_softc*)sc;
39327d5dc18SMarcel Moolenaar struct uart_bas *bas;
394afd396acSMarcel Moolenaar int baudrate, divisor, error;
39527d5dc18SMarcel Moolenaar
39627d5dc18SMarcel Moolenaar bas = &sc->sc_bas;
39706287620SMarcel Moolenaar error = 0;
3988af03381SMarcel Moolenaar uart_lock(sc->sc_hwmtx);
39927d5dc18SMarcel Moolenaar switch (request) {
40027d5dc18SMarcel Moolenaar case UART_IOCTL_BREAK:
40127d5dc18SMarcel Moolenaar if (data)
40227d5dc18SMarcel Moolenaar z8530->tpc |= TPC_BRK;
40327d5dc18SMarcel Moolenaar else
40427d5dc18SMarcel Moolenaar z8530->tpc &= ~TPC_BRK;
40527d5dc18SMarcel Moolenaar uart_setmreg(bas, WR_TPC, z8530->tpc);
40627d5dc18SMarcel Moolenaar uart_barrier(bas);
40727d5dc18SMarcel Moolenaar break;
408afd396acSMarcel Moolenaar case UART_IOCTL_BAUD:
409afd396acSMarcel Moolenaar divisor = uart_getmreg(bas, RR_TCH);
410afd396acSMarcel Moolenaar divisor = (divisor << 8) | uart_getmreg(bas, RR_TCL);
411afd396acSMarcel Moolenaar baudrate = bas->rclk / 2 / (divisor + 2);
412afd396acSMarcel Moolenaar *(int*)data = baudrate;
413afd396acSMarcel Moolenaar break;
41427d5dc18SMarcel Moolenaar default:
41506287620SMarcel Moolenaar error = EINVAL;
41606287620SMarcel Moolenaar break;
41727d5dc18SMarcel Moolenaar }
4188af03381SMarcel Moolenaar uart_unlock(sc->sc_hwmtx);
41906287620SMarcel Moolenaar return (error);
42027d5dc18SMarcel Moolenaar }
42127d5dc18SMarcel Moolenaar
42227d5dc18SMarcel Moolenaar static int
z8530_bus_ipend(struct uart_softc * sc)42327d5dc18SMarcel Moolenaar z8530_bus_ipend(struct uart_softc *sc)
42427d5dc18SMarcel Moolenaar {
425eced4286SMarcel Moolenaar struct z8530_softc *z8530 = (struct z8530_softc*)sc;
42627d5dc18SMarcel Moolenaar struct uart_bas *bas;
42727d5dc18SMarcel Moolenaar int ipend;
42827d5dc18SMarcel Moolenaar uint32_t sig;
429c3c16fcbSMarcel Moolenaar uint8_t bes, ip, iv, src;
43027d5dc18SMarcel Moolenaar
43127d5dc18SMarcel Moolenaar bas = &sc->sc_bas;
43227d5dc18SMarcel Moolenaar ipend = 0;
433c3c16fcbSMarcel Moolenaar
4348af03381SMarcel Moolenaar uart_lock(sc->sc_hwmtx);
435c3c16fcbSMarcel Moolenaar switch (bas->chan) {
436c3c16fcbSMarcel Moolenaar case 1:
437c3c16fcbSMarcel Moolenaar ip = uart_getmreg(bas, RR_IP);
438c3c16fcbSMarcel Moolenaar break;
439c3c16fcbSMarcel Moolenaar case 2: /* XXX hack!!! */
440c3c16fcbSMarcel Moolenaar iv = uart_getmreg(bas, RR_IV) & 0x0E;
441c3c16fcbSMarcel Moolenaar switch (iv) {
442c3c16fcbSMarcel Moolenaar case IV_TEB: ip = IP_TIA; break;
443c3c16fcbSMarcel Moolenaar case IV_XSB: ip = IP_SIA; break;
444c3c16fcbSMarcel Moolenaar case IV_RAB: ip = IP_RIA; break;
445c3c16fcbSMarcel Moolenaar default: ip = 0; break;
446c3c16fcbSMarcel Moolenaar }
447c3c16fcbSMarcel Moolenaar break;
448c3c16fcbSMarcel Moolenaar default:
449c3c16fcbSMarcel Moolenaar ip = 0;
450c3c16fcbSMarcel Moolenaar break;
451c3c16fcbSMarcel Moolenaar }
452c3c16fcbSMarcel Moolenaar
453c3c16fcbSMarcel Moolenaar if (ip & IP_RIA)
4542d511805SMarcel Moolenaar ipend |= SER_INT_RXREADY;
455c3c16fcbSMarcel Moolenaar
456c3c16fcbSMarcel Moolenaar if (ip & IP_TIA) {
457c3c16fcbSMarcel Moolenaar uart_setreg(bas, REG_CTRL, CR_RSTTXI);
458c3c16fcbSMarcel Moolenaar uart_barrier(bas);
459c3c16fcbSMarcel Moolenaar if (z8530->txidle) {
4602d511805SMarcel Moolenaar ipend |= SER_INT_TXIDLE;
4612d511805SMarcel Moolenaar z8530->txidle = 0; /* Mask SER_INT_TXIDLE. */
462c3c16fcbSMarcel Moolenaar }
463c3c16fcbSMarcel Moolenaar }
464c3c16fcbSMarcel Moolenaar
465c3c16fcbSMarcel Moolenaar if (ip & IP_SIA) {
466c3c16fcbSMarcel Moolenaar uart_setreg(bas, REG_CTRL, CR_RSTXSI);
46727d5dc18SMarcel Moolenaar uart_barrier(bas);
46827d5dc18SMarcel Moolenaar bes = uart_getmreg(bas, RR_BES);
469c3c16fcbSMarcel Moolenaar if (bes & BES_BRK)
4702d511805SMarcel Moolenaar ipend |= SER_INT_BREAK;
47127d5dc18SMarcel Moolenaar sig = sc->sc_hwsig;
47228710806SPoul-Henning Kamp SIGCHG(bes & BES_CTS, sig, SER_CTS, SER_DCTS);
47328710806SPoul-Henning Kamp SIGCHG(bes & BES_DCD, sig, SER_DCD, SER_DDCD);
474c3c16fcbSMarcel Moolenaar SIGCHG(bes & BES_SYNC, sig, SER_DSR, SER_DDSR);
475ea549414SMarcel Moolenaar if (sig & SER_MASK_DELTA)
4762d511805SMarcel Moolenaar ipend |= SER_INT_SIGCHG;
47727d5dc18SMarcel Moolenaar src = uart_getmreg(bas, RR_SRC);
47827d5dc18SMarcel Moolenaar if (src & SRC_OVR) {
47927d5dc18SMarcel Moolenaar uart_setreg(bas, REG_CTRL, CR_RSTERR);
480c3c16fcbSMarcel Moolenaar uart_barrier(bas);
4812d511805SMarcel Moolenaar ipend |= SER_INT_OVERRUN;
48227d5dc18SMarcel Moolenaar }
483c3c16fcbSMarcel Moolenaar }
484c3c16fcbSMarcel Moolenaar
485c3c16fcbSMarcel Moolenaar if (ipend) {
486c3c16fcbSMarcel Moolenaar uart_setreg(bas, REG_CTRL, CR_RSTIUS);
487c3c16fcbSMarcel Moolenaar uart_barrier(bas);
488c3c16fcbSMarcel Moolenaar }
489c3c16fcbSMarcel Moolenaar
4908af03381SMarcel Moolenaar uart_unlock(sc->sc_hwmtx);
491c3c16fcbSMarcel Moolenaar
49227d5dc18SMarcel Moolenaar return (ipend);
49327d5dc18SMarcel Moolenaar }
49427d5dc18SMarcel Moolenaar
49527d5dc18SMarcel Moolenaar static int
z8530_bus_param(struct uart_softc * sc,int baudrate,int databits,int stopbits,int parity)49627d5dc18SMarcel Moolenaar z8530_bus_param(struct uart_softc *sc, int baudrate, int databits,
49727d5dc18SMarcel Moolenaar int stopbits, int parity)
49827d5dc18SMarcel Moolenaar {
49927d5dc18SMarcel Moolenaar struct z8530_softc *z8530 = (struct z8530_softc*)sc;
50027d5dc18SMarcel Moolenaar int error;
50127d5dc18SMarcel Moolenaar
5028af03381SMarcel Moolenaar uart_lock(sc->sc_hwmtx);
50327d5dc18SMarcel Moolenaar error = z8530_param(&sc->sc_bas, baudrate, databits, stopbits, parity,
50427d5dc18SMarcel Moolenaar &z8530->tpc);
5058af03381SMarcel Moolenaar uart_unlock(sc->sc_hwmtx);
50627d5dc18SMarcel Moolenaar return (error);
50727d5dc18SMarcel Moolenaar }
50827d5dc18SMarcel Moolenaar
50927d5dc18SMarcel Moolenaar static int
z8530_bus_probe(struct uart_softc * sc)51027d5dc18SMarcel Moolenaar z8530_bus_probe(struct uart_softc *sc)
51127d5dc18SMarcel Moolenaar {
51227d5dc18SMarcel Moolenaar int error;
513875f70dbSMarcel Moolenaar char ch;
51427d5dc18SMarcel Moolenaar
51527d5dc18SMarcel Moolenaar error = z8530_probe(&sc->sc_bas);
51627d5dc18SMarcel Moolenaar if (error)
51727d5dc18SMarcel Moolenaar return (error);
51827d5dc18SMarcel Moolenaar
5194d7abca0SIan Lepore sc->sc_rxfifosz = 3;
5204d7abca0SIan Lepore sc->sc_txfifosz = 1;
5214d7abca0SIan Lepore
522875f70dbSMarcel Moolenaar ch = sc->sc_bas.chan - 1 + 'A';
52327d5dc18SMarcel Moolenaar
524*d73f9fc2SMark Johnston device_set_descf(sc->sc_dev, "z8530, channel %c", ch);
52527d5dc18SMarcel Moolenaar return (0);
52627d5dc18SMarcel Moolenaar }
52727d5dc18SMarcel Moolenaar
52827d5dc18SMarcel Moolenaar static int
z8530_bus_receive(struct uart_softc * sc)52927d5dc18SMarcel Moolenaar z8530_bus_receive(struct uart_softc *sc)
53027d5dc18SMarcel Moolenaar {
53127d5dc18SMarcel Moolenaar struct uart_bas *bas;
53227d5dc18SMarcel Moolenaar int xc;
53327d5dc18SMarcel Moolenaar uint8_t bes, src;
53427d5dc18SMarcel Moolenaar
53527d5dc18SMarcel Moolenaar bas = &sc->sc_bas;
5368af03381SMarcel Moolenaar uart_lock(sc->sc_hwmtx);
53727d5dc18SMarcel Moolenaar bes = uart_getmreg(bas, RR_BES);
53844ed791bSMarcel Moolenaar while (bes & BES_RXA) {
53944ed791bSMarcel Moolenaar if (uart_rx_full(sc)) {
54044ed791bSMarcel Moolenaar sc->sc_rxbuf[sc->sc_rxput] = UART_STAT_OVERRUN;
54144ed791bSMarcel Moolenaar break;
54244ed791bSMarcel Moolenaar }
54327d5dc18SMarcel Moolenaar xc = uart_getreg(bas, REG_DATA);
544c3c16fcbSMarcel Moolenaar uart_barrier(bas);
545c3c16fcbSMarcel Moolenaar src = uart_getmreg(bas, RR_SRC);
54627d5dc18SMarcel Moolenaar if (src & SRC_FE)
54727d5dc18SMarcel Moolenaar xc |= UART_STAT_FRAMERR;
54827d5dc18SMarcel Moolenaar if (src & SRC_PE)
54927d5dc18SMarcel Moolenaar xc |= UART_STAT_PARERR;
550c3c16fcbSMarcel Moolenaar if (src & SRC_OVR)
551c3c16fcbSMarcel Moolenaar xc |= UART_STAT_OVERRUN;
55227d5dc18SMarcel Moolenaar uart_rx_put(sc, xc);
553c3c16fcbSMarcel Moolenaar if (src & (SRC_FE | SRC_PE | SRC_OVR)) {
55427d5dc18SMarcel Moolenaar uart_setreg(bas, REG_CTRL, CR_RSTERR);
55544ed791bSMarcel Moolenaar uart_barrier(bas);
55644ed791bSMarcel Moolenaar }
55744ed791bSMarcel Moolenaar bes = uart_getmreg(bas, RR_BES);
55844ed791bSMarcel Moolenaar }
55944ed791bSMarcel Moolenaar /* Discard everything left in the Rx FIFO. */
56044ed791bSMarcel Moolenaar while (bes & BES_RXA) {
56144ed791bSMarcel Moolenaar (void)uart_getreg(bas, REG_DATA);
562c3c16fcbSMarcel Moolenaar uart_barrier(bas);
563c3c16fcbSMarcel Moolenaar src = uart_getmreg(bas, RR_SRC);
564c3c16fcbSMarcel Moolenaar if (src & (SRC_FE | SRC_PE | SRC_OVR)) {
56544ed791bSMarcel Moolenaar uart_setreg(bas, REG_CTRL, CR_RSTERR);
56644ed791bSMarcel Moolenaar uart_barrier(bas);
56744ed791bSMarcel Moolenaar }
56827d5dc18SMarcel Moolenaar bes = uart_getmreg(bas, RR_BES);
56927d5dc18SMarcel Moolenaar }
5708af03381SMarcel Moolenaar uart_unlock(sc->sc_hwmtx);
57127d5dc18SMarcel Moolenaar return (0);
57227d5dc18SMarcel Moolenaar }
57327d5dc18SMarcel Moolenaar
57427d5dc18SMarcel Moolenaar static int
z8530_bus_setsig(struct uart_softc * sc,int sig)57527d5dc18SMarcel Moolenaar z8530_bus_setsig(struct uart_softc *sc, int sig)
57627d5dc18SMarcel Moolenaar {
57727d5dc18SMarcel Moolenaar struct z8530_softc *z8530 = (struct z8530_softc*)sc;
57827d5dc18SMarcel Moolenaar struct uart_bas *bas;
57927d5dc18SMarcel Moolenaar uint32_t new, old;
58027d5dc18SMarcel Moolenaar
58127d5dc18SMarcel Moolenaar bas = &sc->sc_bas;
58227d5dc18SMarcel Moolenaar do {
58327d5dc18SMarcel Moolenaar old = sc->sc_hwsig;
58427d5dc18SMarcel Moolenaar new = old;
58528710806SPoul-Henning Kamp if (sig & SER_DDTR) {
58628710806SPoul-Henning Kamp SIGCHG(sig & SER_DTR, new, SER_DTR,
58728710806SPoul-Henning Kamp SER_DDTR);
58827d5dc18SMarcel Moolenaar }
58928710806SPoul-Henning Kamp if (sig & SER_DRTS) {
59028710806SPoul-Henning Kamp SIGCHG(sig & SER_RTS, new, SER_RTS,
59128710806SPoul-Henning Kamp SER_DRTS);
59227d5dc18SMarcel Moolenaar }
59327d5dc18SMarcel Moolenaar } while (!atomic_cmpset_32(&sc->sc_hwsig, old, new));
59427d5dc18SMarcel Moolenaar
5958af03381SMarcel Moolenaar uart_lock(sc->sc_hwmtx);
59628710806SPoul-Henning Kamp if (new & SER_DTR)
59727d5dc18SMarcel Moolenaar z8530->tpc |= TPC_DTR;
59827d5dc18SMarcel Moolenaar else
59927d5dc18SMarcel Moolenaar z8530->tpc &= ~TPC_DTR;
60028710806SPoul-Henning Kamp if (new & SER_RTS)
60127d5dc18SMarcel Moolenaar z8530->tpc |= TPC_RTS;
60227d5dc18SMarcel Moolenaar else
60327d5dc18SMarcel Moolenaar z8530->tpc &= ~TPC_RTS;
60427d5dc18SMarcel Moolenaar uart_setmreg(bas, WR_TPC, z8530->tpc);
60527d5dc18SMarcel Moolenaar uart_barrier(bas);
6068af03381SMarcel Moolenaar uart_unlock(sc->sc_hwmtx);
60727d5dc18SMarcel Moolenaar return (0);
60827d5dc18SMarcel Moolenaar }
60927d5dc18SMarcel Moolenaar
61027d5dc18SMarcel Moolenaar static int
z8530_bus_transmit(struct uart_softc * sc)61127d5dc18SMarcel Moolenaar z8530_bus_transmit(struct uart_softc *sc)
61227d5dc18SMarcel Moolenaar {
613eced4286SMarcel Moolenaar struct z8530_softc *z8530 = (struct z8530_softc*)sc;
61427d5dc18SMarcel Moolenaar struct uart_bas *bas;
61527d5dc18SMarcel Moolenaar
61627d5dc18SMarcel Moolenaar bas = &sc->sc_bas;
6178af03381SMarcel Moolenaar uart_lock(sc->sc_hwmtx);
61827d5dc18SMarcel Moolenaar while (!(uart_getmreg(bas, RR_BES) & BES_TXE))
61927d5dc18SMarcel Moolenaar ;
62027d5dc18SMarcel Moolenaar uart_setreg(bas, REG_DATA, sc->sc_txbuf[0]);
62127d5dc18SMarcel Moolenaar uart_barrier(bas);
62227d5dc18SMarcel Moolenaar sc->sc_txbusy = 1;
6232d511805SMarcel Moolenaar z8530->txidle = 1; /* Report SER_INT_TXIDLE again. */
6248af03381SMarcel Moolenaar uart_unlock(sc->sc_hwmtx);
62527d5dc18SMarcel Moolenaar return (0);
62627d5dc18SMarcel Moolenaar }
627d76a1ef4SWarner Losh
628d76a1ef4SWarner Losh static void
z8530_bus_grab(struct uart_softc * sc)629d76a1ef4SWarner Losh z8530_bus_grab(struct uart_softc *sc)
630d76a1ef4SWarner Losh {
631d76a1ef4SWarner Losh struct uart_bas *bas;
632d76a1ef4SWarner Losh
633d76a1ef4SWarner Losh bas = &sc->sc_bas;
634d76a1ef4SWarner Losh uart_lock(sc->sc_hwmtx);
635d76a1ef4SWarner Losh uart_setmreg(bas, WR_IDT, IDT_XIE | IDT_TIE);
636d76a1ef4SWarner Losh uart_barrier(bas);
637d76a1ef4SWarner Losh uart_unlock(sc->sc_hwmtx);
638d76a1ef4SWarner Losh }
639d76a1ef4SWarner Losh
640d76a1ef4SWarner Losh static void
z8530_bus_ungrab(struct uart_softc * sc)641d76a1ef4SWarner Losh z8530_bus_ungrab(struct uart_softc *sc)
642d76a1ef4SWarner Losh {
643d76a1ef4SWarner Losh struct uart_bas *bas;
644d76a1ef4SWarner Losh
645d76a1ef4SWarner Losh bas = &sc->sc_bas;
646d76a1ef4SWarner Losh uart_lock(sc->sc_hwmtx);
647d76a1ef4SWarner Losh uart_setmreg(bas, WR_IDT, IDT_XIE | IDT_TIE | IDT_RIA);
648d76a1ef4SWarner Losh uart_barrier(bas);
649d76a1ef4SWarner Losh uart_unlock(sc->sc_hwmtx);
650d76a1ef4SWarner Losh }
651