xref: /freebsd-14.2/sys/dev/uart/uart_dev_z8530.c (revision d73f9fc2)
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