summaryrefslogtreecommitdiff
path: root/sys/arch/mac68k/dev
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch/mac68k/dev')
-rw-r--r--sys/arch/mac68k/dev/adb_direct.c14
-rw-r--r--sys/arch/mac68k/dev/pm_direct.c9
-rw-r--r--sys/arch/mac68k/dev/z8530reg.h51
-rw-r--r--sys/arch/mac68k/dev/z8530sc.c152
-rw-r--r--sys/arch/mac68k/dev/z8530sc.h128
-rw-r--r--sys/arch/mac68k/dev/z8530tty.c764
-rw-r--r--sys/arch/mac68k/dev/z8530tty.h140
-rw-r--r--sys/arch/mac68k/dev/zs.c905
8 files changed, 1045 insertions, 1118 deletions
diff --git a/sys/arch/mac68k/dev/adb_direct.c b/sys/arch/mac68k/dev/adb_direct.c
index 3200c91a940..73214a76ef9 100644
--- a/sys/arch/mac68k/dev/adb_direct.c
+++ b/sys/arch/mac68k/dev/adb_direct.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: adb_direct.c,v 1.13 2003/11/03 06:43:02 david Exp $ */
+/* $OpenBSD: adb_direct.c,v 1.14 2004/11/25 18:32:10 miod Exp $ */
/* $NetBSD: adb_direct.c,v 1.5 1997/04/21 18:04:28 scottr Exp $ */
/* From: adb_direct.c 2.02 4/18/97 jpw */
@@ -204,8 +204,6 @@ int ADBNumDevices; /* num. of ADB devices found with ADBReInit */
extern struct mac68k_machine_S mac68k_machine;
-int zshard(int);
-
void pm_setup_adb(void);
void pm_check_adb_devices(int);
void pm_intr(void);
@@ -1073,7 +1071,7 @@ switch_start:
ADB_SET_STATE_ACKON(); /* start ACK to ADB chip */
delay(ADB_DELAY); /* delay */
ADB_SET_STATE_ACKOFF(); /* end ACK to ADB chip */
- zshard(0); /* grab any serial interrupts */
+ (void)intr_dispatch(0x70);
break;
case ADB_ACTION_IN:
@@ -1087,7 +1085,7 @@ switch_start:
ADB_SET_STATE_ACKON(); /* start ACK to ADB chip */
delay(ADB_DELAY); /* delay */
ADB_SET_STATE_ACKOFF(); /* end ACK to ADB chip */
- zshard(0); /* grab any serial interrupts */
+ (void)intr_dispatch(0x70);
if (1 == ending) { /* end of message? */
ADB_SET_STATE_INACTIVE(); /* signal end of frame */
@@ -1148,7 +1146,7 @@ switch_start:
adbActionState = ADB_ACTION_OUT; /* set next state */
delay(ADB_DELAY); /* delay */
- zshard(0); /* grab any serial interrupts */
+ (void)intr_dispatch(0x70);
if (ADB_INTR_IS_ON) { /* ADB intr low during
* write */
@@ -1189,13 +1187,13 @@ switch_start:
adbWriteDelay = 1; /* must retry when done with
* read */
delay(ADB_DELAY); /* delay */
- zshard(0); /* grab any serial interrupts */
+ (void)intr_dispatch(0x70);
goto switch_start; /* process next state right
* now */
break;
}
delay(ADB_DELAY); /* required delay */
- zshard(0); /* grab any serial interrupts */
+ (void)intr_dispatch(0x70);
if (adbOutputBuffer[0] == adbSentChars) { /* check for done */
if (0 == adb_cmd_result(adbOutputBuffer)) { /* do we expect data
diff --git a/sys/arch/mac68k/dev/pm_direct.c b/sys/arch/mac68k/dev/pm_direct.c
index 9602a3e5741..3ed9b2ae26e 100644
--- a/sys/arch/mac68k/dev/pm_direct.c
+++ b/sys/arch/mac68k/dev/pm_direct.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: pm_direct.c,v 1.6 2003/03/14 10:47:36 miod Exp $ */
+/* $OpenBSD: pm_direct.c,v 1.7 2004/11/25 18:32:10 miod Exp $ */
/* pm_direct.c 1.22 01/09/97 Takashi Hamada */
/*
@@ -173,7 +173,6 @@ extern int adbStarting; /* doing ADB reinit, so do "polling" differently */
/*
* Define the external functions
*/
-extern int zshard(int); /* from zs.c */
extern void adb_comp_exec(void); /* from adb_direct.c */
@@ -255,7 +254,7 @@ pm_wait_busy(delay)
{
while(PM_IS_ON) {
#ifdef PM_GRAB_SI
- zshard(0); /* grab any serial interrupts */
+ (void)intr_dispatch(0x70);
#endif
if ((--delay) < 0)
return( 1 ); /* timeout */
@@ -273,7 +272,7 @@ pm_wait_free(delay)
{
while(PM_IS_OFF) {
#ifdef PM_GRAB_SI
- zshard(0); /* grab any serial interrupts */
+ (void)intr_dispatch(0x70);
#endif
if ((--delay) < 0)
return( 0 ); /* timeout */
@@ -980,7 +979,7 @@ pm_adb_op(buffer, compRout, data, command)
if ((via_reg(VIA1, vIFR) & 0x10) == 0x10)
pm_intr();
#ifdef PM_GRAB_SI
- zshard(0); /* grab any serial interrupts */
+ (void)intr_dispatch(0x70);
#endif
if ((--delay) < 0)
return( -1 );
diff --git a/sys/arch/mac68k/dev/z8530reg.h b/sys/arch/mac68k/dev/z8530reg.h
index cc3224ba9d5..57a1cc636cf 100644
--- a/sys/arch/mac68k/dev/z8530reg.h
+++ b/sys/arch/mac68k/dev/z8530reg.h
@@ -1,5 +1,5 @@
-/* $OpenBSD: z8530reg.h,v 1.4 2003/06/02 23:27:49 millert Exp $ */
-/* $NetBSD: z8530reg.h,v 1.3 1996/05/18 18:54:23 briggs Exp $ */
+/* $OpenBSD: z8530reg.h,v 1.5 2004/11/25 18:32:10 miod Exp $ */
+/* $NetBSD: z8530reg.h,v 1.7 1996/10/23 00:32:31 gwr Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -60,29 +60,12 @@
* registers 2 and 9 across both channels, and reads registers 2 and 3
* differently for the two channels. We can, however, ignore this much
* of the time.
- */
-
-/*
+ *
* This file also includes flags for the Z85C30 and Z85230 enhanced scc.
* The CMOS 8530 includes extra SDLC functionality, and is used in a
* number of Macs (often in the Z85C80, an 85C30 combined w/ a SCSI
* controller). -wrs
- */
-#if 0 /* Example only! */
-/*
- * The layout of this structure is hardware-dependent!
- * Define these in some machine-dependent place.
- */
-struct zschan {
- volatile u_char zc_csr; /* ctrl, status, or reg. number */
- volatile u_char zc_data; /* data or numbered register */
-};
-struct zsdevice {
- struct zschan zs_chan[2];
-};
-#endif /* Example only! */
-
-/*
+ *
* Some of the names in this files were chosen to make the hsis driver
* work unchanged (which means that they will match some in SunOS).
*
@@ -91,9 +74,7 @@ struct zsdevice {
* framing error (missing stop bit, etc)
* end of frame (in synchronous modes)
* parity error (when `parity error is S.C.' is set)
- */
-
-/*
+ *
* Registers with only a single `numeric value' get a name.
* Other registers hold bits and are only numbered; the bit
* definitions imply the register number (see below).
@@ -181,6 +162,8 @@ struct zsdevice {
#define ZSWR1_TIE 0x02 /* transmit interrupt enable */
#define ZSWR1_SIE 0x01 /* external/status interrupt enable */
+#define ZSWR1_IMASK 0x1F /* mask of all itr. enable bits. */
+
/* HSIS compat */
#define ZSWR1_REQ_ENABLE (ZSWR1_REQ_WAIT | ZSWR1_REQ_TX)
@@ -197,6 +180,7 @@ struct zsdevice {
#define ZSWR3_RX_7 0x40 /* receive 7 bits per char */
#define ZSWR3_RX_6 0x80 /* receive 6 bits per char */
#define ZSWR3_RX_8 0xc0 /* receive 8 bits per char */
+#define ZSWR3_RXSIZE 0xc0 /* receive char size mask */
#define ZSWR3_HFC 0x20 /* hardware flow control */
#define ZSWR3_HUNT 0x10 /* enter hunt mode */
@@ -213,19 +197,23 @@ struct zsdevice {
#define ZSWR4_CLK_X16 0x40 /* clock divisor = 16 */
#define ZSWR4_CLK_X32 0x80 /* clock divisor = 32 */
#define ZSWR4_CLK_X64 0xc0 /* clock divisor = 64 */
+#define ZSWR4_CLK_MASK 0xc0 /* clock divisor mask */
#define ZSWR4_MONOSYNC 0x00 /* 8 bit sync char (sync only) */
#define ZSWR4_BISYNC 0x10 /* 16 bit sync char (sync only) */
-#define ZSWR4_SDLC 0x20 /* SDLC mode */
+#define ZSWR4_SDLC 0x20 /* SDLC mode */
#define ZSWR4_EXTSYNC 0x30 /* external sync mode */
+#define ZSWR4_SYNC_MASK 0x30 /* sync mode bit mask */
-#define ZSWR4_SYNCMODE 0x00 /* one of the above sync modes */
-#define ZSWR4_ONESB 0x04 /* 1 stop bit */
-#define ZSWR4_1P5SB 0x08 /* 1.5 stop bits (clk cannot be 1x) */
-#define ZSWR4_TWOSB 0x0c /* 2 stop bits */
+#define ZSWR4_SYNCMODE 0x00 /* no stop bit (sync mode only) */
+#define ZSWR4_ONESB 0x04 /* 1 stop bit */
+#define ZSWR4_1P5SB 0x08 /* 1.5 stop bits (clk cannot be 1x) */
+#define ZSWR4_TWOSB 0x0c /* 2 stop bits */
+#define ZSWR4_SBMASK 0x0c /* mask of all stop bits */
-#define ZSWR4_EVENP 0x02 /* check for even parity */
+#define ZSWR4_EVENP 0x02 /* check for even parity */
#define ZSWR4_PARENB 0x01 /* enable parity checking */
+#define ZSWR4_PARMASK 0x03 /* mask of all parity bits */
/*
* Bits in Write Register 5 (`Transmit Parameter and Controls').
@@ -238,6 +226,7 @@ struct zsdevice {
#define ZSWR5_TX_7 0x20 /* transmit 7 bits */
#define ZSWR5_TX_6 0x40 /* transmit 6 bits */
#define ZSWR5_TX_8 0x60 /* transmit 8 bits */
+#define ZSWR5_TXSIZE 0x60 /* transmit char size mask */
#define ZSWR5_BREAK 0x10 /* send break (continuous 0s) */
#define ZSWR5_TX_ENABLE 0x08 /* enable transmitter */
@@ -340,7 +329,7 @@ struct zsdevice {
* 2 bps
*
* rounded down to an integer. This can be computed entirely
- * in integer arithemtic as:
+ * in integer arithmetic as:
*
* f + bps
* ------- - 2
diff --git a/sys/arch/mac68k/dev/z8530sc.c b/sys/arch/mac68k/dev/z8530sc.c
index acded984c68..6ec3fef5ac6 100644
--- a/sys/arch/mac68k/dev/z8530sc.c
+++ b/sys/arch/mac68k/dev/z8530sc.c
@@ -1,5 +1,5 @@
-/* $OpenBSD: z8530sc.c,v 1.6 2004/08/03 12:10:47 todd Exp $ */
-/* $NetBSD: z8530sc.c,v 1.1 1996/05/18 18:54:28 briggs Exp $ */
+/* $OpenBSD: z8530sc.c,v 1.7 2004/11/25 18:32:10 miod Exp $ */
+/* $NetBSD: z8530sc.c,v 1.5 1996/12/17 20:42:40 gwr Exp $ */
/*
* Copyright (c) 1994 Gordon W. Ross
@@ -61,11 +61,13 @@
#include <sys/kernel.h>
#include <sys/syslog.h>
-/* #include <dev/ic/z8530reg.h> */
-#include "z8530reg.h"
+#include <mac68k/dev/z8530reg.h>
#include <machine/z8530var.h>
-int
+static void zsnull_intr(struct zs_chanstate *);
+static void zsnull_softint(struct zs_chanstate *);
+
+void
zs_break(cs, set)
struct zs_chanstate *cs;
int set;
@@ -82,26 +84,10 @@ zs_break(cs, set)
}
zs_write_reg(cs, 5, cs->cs_creg[5]);
splx(s);
-
- return 0;
}
/*
- * Compute the current baud rate given a ZSCC channel.
- */
-int
-zs_getspeed(cs)
- struct zs_chanstate *cs;
-{
- int tconst;
-
- tconst = zs_read_reg(cs, 12);
- tconst |= zs_read_reg(cs, 13) << 8;
- return (TCONST_TO_BPS(cs->cs_pclk_div16, tconst));
-}
-
-/*
* drain on-chip fifo
*/
void
@@ -130,49 +116,6 @@ zs_iflush(cs)
}
}
-/*
- * Figure out if a chip is an NMOS 8530, a CMOS 8530,
- * or an 85230. We use a form of the test in the Zilog SCC
- * users manual.
- */
-int
-zs_checkchip(cs)
- struct zs_chanstate *cs;
-{
- char r1, r2, r3;
- int chip;
-
- /* we assume we can write to the chip */
-
- r1=cs->cs_creg[15]; /* see if bit 0 sticks */
- zs_write_reg(cs, 15, (r1 | ZSWR15_ENABLE_ENHANCED));
- if ((zs_read_reg(cs, 15) & ZSWR15_ENABLE_ENHANCED) != 0) {
- /* we have either an 8580 or 85230. NB Zilog says we should only
- * have an 85230 at this point, but the 8580 seems to pass this
- * test too. To test, we try to write to WR7', and see if we
- * loose sight of RR14. */
- r2=cs->cs_creg[14];
- r3=(r2 != 0x47) ? 0x47 : 0x40;
- /* unique bit pattern to turn on reading of WR7' at RR14 */
- zs_write_reg(cs, 7, ~r2);
- if (zs_read_reg(cs, ZSRR_ENHANCED) != r2) {
- chip = ZS_CHIP_ESCC;
- zs_write_reg(cs, 7, cs->cs_creg[ZS_ENHANCED_REG]);
- } else {
- chip = ZS_CHIP_8580;
- zs_write_reg(cs, 7, cs->cs_creg[7]);
- }
- zs_write_reg(cs, 15, r1);
- } else { /* now we have to tell an NMOS from a CMOS; does WR15 D2 work? */
- zs_write_reg(cs, 15, (r1 | ZSWR15_SDLC_FIFO));
- r2=cs->cs_creg[2];
- zs_write_reg(cs, 2, (r2 | 0x80));
- chip = (zs_read_reg(cs, 6) & 0x80) ? ZS_CHIP_NMOS : ZS_CHIP_CMOS;
- zs_write_reg(cs, 2, r2);
- }
- zs_write_reg(cs, 15, r1);
- return chip;
-}
/*
* Write the given register set to the given zs channel in the proper order.
@@ -201,8 +144,7 @@ zs_loadchannelregs(cs)
#endif
/* disable interrupts */
- zs_write_reg(cs, 1, reg[1] &
- ~(ZSWR1_RIE_SPECIAL_ONLY | ZSWR1_TIE | ZSWR1_SIE));
+ zs_write_reg(cs, 1, reg[1] & ~ZSWR1_IMASK);
/* baud clock divisor, stop bits, parity */
zs_write_reg(cs, 4, reg[4]);
@@ -234,11 +176,11 @@ zs_loadchannelregs(cs)
/* Shut down the BRG */
zs_write_reg(cs, 14, reg[14] & ~ZSWR14_BAUD_ENA);
-
- if ((cs->cs_cclk_flag & ZSC_EXTERN) ||
- (cs->cs_pclk_flag & ZSC_EXTERN))
- zsmd_setclock(cs);
- /* the md layer wants to do something; let it. */
+
+#ifdef ZS_MD_SETCLK
+ /* Let the MD code setup any external clock. */
+ ZS_MD_SETCLK(cs);
+#endif /* ZS_MD_SETCLK */
/* clock mode control */
zs_write_reg(cs, 11, reg[11]);
@@ -253,7 +195,8 @@ zs_loadchannelregs(cs)
/* which lines cause status interrupts */
zs_write_reg(cs, 15, reg[15]);
- /* Zilog docs recommend resetting external status twice at this
+ /*
+ * Zilog docs recommend resetting external status twice at this
* point. Mainly as the status bits are latched, and the first
* interrupt clear might unlatch them to new values, generating
* a second interrupt request.
@@ -265,11 +208,8 @@ zs_loadchannelregs(cs)
zs_write_reg(cs, 3, reg[3]);
zs_write_reg(cs, 5, reg[5]);
- /* interrupt enables: TX, TX, STATUS */
+ /* interrupt enables: RX, TX, STATUS */
zs_write_reg(cs, 1, reg[1]);
-
- cs->cs_cclk_flag = cs->cs_pclk_flag;
- cs->cs_csource = cs->cs_psource;
}
@@ -293,53 +233,37 @@ zsc_intr_hard(arg)
register struct zs_chanstate *cs_b;
register int rval;
register u_char rr3, rr3a;
-#ifdef DIAGNOSTIC
- register int loopcount;
- loopcount = ZS_INTERRUPT_CNT;
-#endif
- cs_a = &zsc->zsc_cs[0];
- cs_b = &zsc->zsc_cs[1];
+ cs_a = zsc->zsc_cs[0];
+ cs_b = zsc->zsc_cs[1];
rval = 0;
rr3a = 0;
/* Note: only channel A has an RR3 */
- rr3 = zs_read_reg(cs_a, 3);
-
- while ((rr3 = zs_read_reg(cs_a, ZSRR_IPEND))
-#ifdef DIAGNOSTIC
- && --loopcount
-#endif
- ) {
+ while ((rr3 = zs_read_reg(cs_a, 3)) != 0) {
/* Handle receive interrupts first. */
if (rr3 & ZSRR3_IP_A_RX)
(*cs_a->cs_ops->zsop_rxint)(cs_a);
if (rr3 & ZSRR3_IP_B_RX)
- (*cs_b->cs_ops->zsop_rxint)(cs_b);
+ (*cs_b->cs_ops->zsop_stint)(cs_b);
/* Handle status interrupts (i.e. flow control). */
if (rr3 & ZSRR3_IP_A_STAT)
(*cs_a->cs_ops->zsop_stint)(cs_a);
if (rr3 & ZSRR3_IP_B_STAT)
(*cs_b->cs_ops->zsop_stint)(cs_b);
-
+
/* Handle transmit done interrupts. */
if (rr3 & ZSRR3_IP_A_TX)
(*cs_a->cs_ops->zsop_txint)(cs_a);
if (rr3 & ZSRR3_IP_B_TX)
(*cs_b->cs_ops->zsop_txint)(cs_b);
-
+
+ /* Accumulate so we know what needs to be cleared. */
rr3a |= rr3;
}
-#ifdef DIAGNOSTIC
- if (loopcount == 0) {
- if (rr3 & (ZSRR3_IP_A_RX | ZSRR3_IP_A_TX | ZSRR3_IP_A_STAT))
- cs_a->cs_flags |= ZS_FLAGS_INTERRUPT_OVERRUN;
- if (rr3 & (ZSRR3_IP_B_RX | ZSRR3_IP_B_TX | ZSRR3_IP_B_STAT))
- cs_b->cs_flags |= ZS_FLAGS_INTERRUPT_OVERRUN;
- }
-#endif
+
/* Clear interrupt. */
if (rr3a & (ZSRR3_IP_A_RX | ZSRR3_IP_A_TX | ZSRR3_IP_A_STAT)) {
@@ -351,11 +275,7 @@ zsc_intr_hard(arg)
rval |= 2;
}
- if ((cs_a->cs_softreq) || (cs_b->cs_softreq)) {
- /* This is a machine-dependent function (or macro). */
- zsc_req_softint(zsc);
- }
-
+ /* Note: caller will check cs_x->cs_softreq and DTRT. */
return (rval);
}
@@ -369,11 +289,11 @@ zsc_intr_soft(arg)
{
register struct zsc_softc *zsc = arg;
register struct zs_chanstate *cs;
- register int rval, unit;
+ register int rval, chan;
rval = 0;
- for (unit = 0; unit < 2; unit++) {
- cs = &zsc->zsc_cs[unit];
+ for (chan = 0; chan < 2; chan++) {
+ cs = zsc->zsc_cs[chan];
/*
* The softint flag can be safely cleared once
@@ -383,27 +303,33 @@ zsc_intr_soft(arg)
if (cs->cs_softreq) {
cs->cs_softreq = 0;
(*cs->cs_ops->zsop_softint)(cs);
- rval = 1;
+ rval++;
}
}
return (rval);
}
-static void zsnull_intr(struct zs_chanstate *);
-static void zsnull_softint(struct zs_chanstate *);
+/*
+ * Provide a null zs "ops" vector.
+ */
+
+static void zsnull_intr(struct zs_chanstate *);
+static void zsnull_softint(struct zs_chanstate *);
static void
zsnull_intr(cs)
struct zs_chanstate *cs;
{
- zs_write_reg(cs, 1, 0);
- zs_write_reg(cs, 15, 0);
+ /* Ask for softint() call. */
+ cs->cs_softreq = 1;
}
static void
zsnull_softint(cs)
struct zs_chanstate *cs;
{
+ zs_write_reg(cs, 1, 0);
+ zs_write_reg(cs, 15, 0);
}
struct zsops zsops_null = {
diff --git a/sys/arch/mac68k/dev/z8530sc.h b/sys/arch/mac68k/dev/z8530sc.h
index cbde3c453ed..6604c744bff 100644
--- a/sys/arch/mac68k/dev/z8530sc.h
+++ b/sys/arch/mac68k/dev/z8530sc.h
@@ -1,5 +1,5 @@
-/* $OpenBSD: z8530sc.h,v 1.5 2003/06/02 23:27:49 millert Exp $ */
-/* $NetBSD: z8530sc.h,v 1.1 1996/05/18 18:54:30 briggs Exp $ */
+/* $OpenBSD: z8530sc.h,v 1.6 2004/11/25 18:32:10 miod Exp $ */
+/* $NetBSD: z8530sc.h,v 1.5 1996/12/17 20:42:42 gwr Exp $ */
/*
* Copyright (c) 1994 Gordon W. Ross
@@ -44,33 +44,6 @@
/*
- * Clock source info structure
- */
-struct zsclksrc {
- long clk; /* clock rate, in MHz, present on signal line */
- int flags; /* Specifies how this source can be used
- (RTxC divided, RTxC BRG, PCLK BRG, TRxC divided)
- and also if the source is "external" and if it
- is changeable (by an ioctl ex.). The
- source usage flags are used by the tty
- child. The other bits tell zsloadchannelregs
- if it should call an md signal source
- changing routine. ZSC_VARIABLE says if
- an ioctl should be able to cahnge the
- clock rate.*/
-};
-#define ZSC_PCLK 0x01
-#define ZSC_RTXBRG 0x02
-#define ZSC_RTXDIV 0x04
-#define ZSC_TRXDIV 0x08
-#define ZSC_VARIABLE 0x40
-#define ZSC_EXTERN 0x80
-
-#define ZSC_BRG 0x03
-#define ZSC_DIV 0x0c
-
-
-/*
* Software state, per zs channel.
*/
struct zs_chanstate {
@@ -80,16 +53,13 @@ struct zs_chanstate {
volatile u_char *cs_reg_data; /* data or numbered register */
int cs_channel; /* sub-unit number */
- void *cs_private; /* sub-driver data pointer */
+ void *cs_private; /* sub-driver data pointer */
struct zsops *cs_ops;
- int cs_clock_count; /* how many signal sources available */
- struct zsclksrc cs_clocks[4]; /* info on available signal sources */
- int cs_defspeed; /* default baud rate (from PROM) */
- int cs_pclk_div16; /* PCLK / 16 used by zs children w/o multiple
- * baud support - some ports have only
- * one clock source, and some children (kbd & ms)
- * are fixed baud rate */
+ int cs_brg_clk; /* BAUD Rate Generator clock
+ * (usually PCLK / 16) */
+ int cs_defspeed; /* default baud rate */
+ int cs_defcflag; /* default cflag */
/*
* We must keep a copy of the write registers as they are
@@ -106,38 +76,28 @@ struct zs_chanstate {
*/
u_char cs_creg[16]; /* current values */
u_char cs_preg[16]; /* pending values */
- long cs_cclk_flag; /* flag for current clock source */
- long cs_pclk_flag; /* flag for pending clock source */
- int cs_csource; /* current source # */
- int cs_psource; /* pending source # */
+ int cs_heldchange; /* change pending (creg != preg) */
- u_char cs_heldchange; /* change pending (creg != preg) */
u_char cs_rr0; /* last rr0 processed */
- u_char cs_rr0_changes; /* rr0 changes noted in status int. */
+ u_char cs_rr0_delta; /* rr0 changes at status intr. */
+ u_char cs_rr0_dcd; /* which bit to read as DCD */
+ u_char cs_rr0_cts; /* which bit to read as CTS */
+ /* the above is set only while CRTSCTS is enabled. */
+
+ u_char cs_wr5_dtr; /* which bit to write as DTR */
+ u_char cs_wr5_rts; /* which bit to write as RTS */
+ /* the above is set only while CRTSCTS is enabled. */
char cs_softreq; /* need soft interrupt call */
- char cs_chip; /* type of chip */
- char cs_flags; /* misc. flags */
+ char cs_pad[1];
+ /* MD code might define a larger variant of this. */
};
-#define ZS_INTERRUPT_CNT 10
-#define ZS_FLAGS_INTERRUPT_OVERRUN 0x01
-#define ZS_FLAGS_DEBUG1 0x10
- /* The interrupt service routine will now look to see if more interrupts
- * come in while servicing an interrupt. If so, it keeps servicing
- * them either to exhaustion or until it's tried ZS_INTERRUPT_CNT times.
- * If it tries too many times, it flags a ZS_FLAGS_INTERRUPT_OVERRUN.
- * It shouldn't, but w/o a counter of sorts, we could get hung in an
- * infinite loop because of sick hardware, or because there's a data
- * clock fed on one of the inputs. The DEBUG flag is for testing and has
- * no permanant definition. */
-#define ZS_ENHANCED_REG 8
- /* cs_Xreg which is used to hold WR7' data; reg 8 is an alias to the
- * data port, so we won't miss its loss. */
/*
* Function vector - per channel
*/
-typedef void (*zsop_t)(register struct zs_chanstate *);
+struct zs_chanstate;
+typedef void (*zsop_t) (struct zs_chanstate *);
struct zsops {
zsop_t zsop_rxint; /* receive char available */
zsop_t zsop_stint; /* external/status */
@@ -147,38 +107,24 @@ struct zsops {
extern struct zsops zsops_null;
-struct zsc_softc {
- struct device zsc_dev; /* required first: base device */
- struct zs_chanstate zsc_cs[2]; /* channel A and B soft state */
-};
-
struct zsc_attach_args {
int channel; /* two serial channels per zsc */
int hwflags;
};
-#define ZS_HWFLAG_CONSOLE 1
-#define ZS_HWFLAG_CONABRT 2
-#define ZS_HWFLAG_RAW 4
-#define ZS_HWFLAG_IGCTS 16
-#define ZS_HWFLAG_IGDCD 32
-/* _CONSOLE says this port is the console, _CONABRT says a Break sequence acts as
- an abort, and _RAW recomends "raw" mode defaults on a tty.
- _CONABRT is turned off if an overly-long break is received.
- _IGCTS and _IGDCD tell the tty layer to ignore CTS or DCD. Assume
- whatever's least supprising (CTS and DCD present). Used mainly for
- external clock support on mac68k. The DCD and CTS pins are used also
- for clock inputs; not good for the UNIX I/O model! */
-
-#define ZS_CHIP_NMOS 0
-#define ZS_CHIP_CMOS 1
-#define ZS_CHIP_8580 2
-#define ZS_CHIP_ESCC 3
-
-void zs_loadchannelregs(struct zs_chanstate *);
-int zsc_intr_soft(void *);
-int zsc_intr_hard(void *);
-int zs_checkchip(struct zs_chanstate *);
-int zs_break(struct zs_chanstate *, int);
-int zs_getspeed(struct zs_chanstate *);
-void zs_iflush(struct zs_chanstate *);
-
+#define ZS_HWFLAG_CONSOLE 1
+#define ZS_HWFLAG_NO_DCD 2 /* Ignore the DCD bit */
+#define ZS_HWFLAG_NO_CTS 4 /* Ignore the CTS bit */
+#define ZS_HWFLAG_RAW 8 /* advise raw mode */
+
+int zsc_intr_hard(void *);
+int zsc_intr_soft(void *);
+
+void zs_abort(struct zs_chanstate *);
+void zs_break(struct zs_chanstate *, int);
+void zs_iflush (struct zs_chanstate *);
+void zs_loadchannelregs(struct zs_chanstate *);
+int zs_set_speed (struct zs_chanstate *, int);
+int zs_set_modes (struct zs_chanstate *, int);
+int zs_getspeed(struct zs_chanstate *);
+
+extern int zs_major;
diff --git a/sys/arch/mac68k/dev/z8530tty.c b/sys/arch/mac68k/dev/z8530tty.c
index 975e36851b8..e2bc82c2f67 100644
--- a/sys/arch/mac68k/dev/z8530tty.c
+++ b/sys/arch/mac68k/dev/z8530tty.c
@@ -1,5 +1,5 @@
-/* $OpenBSD: z8530tty.c,v 1.15 2003/11/03 06:43:02 david Exp $ */
-/* $NetBSD: z8530tty.c,v 1.10 1996/12/18 05:17:44 scottr Exp $ */
+/* $OpenBSD: z8530tty.c,v 1.16 2004/11/25 18:32:10 miod Exp $ */
+/* $NetBSD: z8530tty.c,v 1.14 1996/12/17 20:42:43 gwr Exp $ */
/*
* Copyright (c) 1994 Gordon W. Ross
@@ -47,6 +47,19 @@
*
* This is the "slave" driver that will be attached to
* the "zsc" driver for plain "tty" async. serial lines.
+ *
+ * Credits, history:
+ *
+ * The original version of this code was the sparc/dev/zs.c driver
+ * as distributed with the Berkeley 4.4 Lite release. Since then,
+ * Gordon Ross reorganized the code into the current parent/child
+ * driver scheme, separating the Sun keyboard and mouse support
+ * into independent child drivers.
+ *
+ * RTS/CTS flow-control support was a collaboration of:
+ * Gordon Ross <gwr@netbsd.org>,
+ * Bill Studenmund <wrstuden@loki.stanford.edu>
+ * Ian Dall <Ian.Dall@dsto.defence.gov.au>
*/
#include <sys/param.h>
@@ -62,8 +75,7 @@
#include <sys/kernel.h>
#include <sys/syslog.h>
-/* #include <dev/ic/z8530reg.h> */
-#include "z8530reg.h"
+#include <mac68k/dev/z8530reg.h>
#include <machine/z8530var.h>
#ifdef KGDB
@@ -71,6 +83,15 @@ extern int zs_check_kgdb();
#endif
/*
+ * How many input characters we can buffer.
+ * The port-specific var.h may override this.
+ * Note: must be a power of two!
+ */
+#ifndef ZSTTY_RING_SIZE
+#define ZSTTY_RING_SIZE 2048
+#endif
+
+/*
* Make this an option variable one can patch.
* But be warned: this must be a power of 2!
*/
@@ -79,7 +100,50 @@ int zstty_rbuf_size = ZSTTY_RING_SIZE;
/* This should usually be 3/4 of ZSTTY_RING_SIZE */
int zstty_rbuf_hiwat = (ZSTTY_RING_SIZE - (ZSTTY_RING_SIZE >> 2));
-struct zstty_stats z8530tty_stats;
+struct zstty_softc {
+ struct device zst_dev; /* required first: base device */
+ struct tty *zst_tty;
+ struct zs_chanstate *zst_cs;
+
+ int zst_hwflags; /* see z8530var.h */
+ int zst_swflags; /* TIOCFLAG_SOFTCAR, ... <ttycom.h> */
+
+ /*
+ * Printing an overrun error message often takes long enough to
+ * cause another overrun, so we only print one per second.
+ */
+ long zst_rotime; /* time of last ring overrun */
+ long zst_fotime; /* time of last fifo overrun */
+
+ /*
+ * The receive ring buffer.
+ */
+ int zst_rbget; /* ring buffer `get' index */
+ volatile int zst_rbput; /* ring buffer `put' index */
+ int zst_ringmask;
+ int zst_rbhiwat;
+
+ u_short *zst_rbuf; /* rr1, data pairs */
+
+ /*
+ * The transmit byte count and address are used for pseudo-DMA
+ * output in the hardware interrupt code. PDMA can be suspended
+ * to get pending changes done; heldtbc is used for this. It can
+ * also be stopped for ^S; this sets TS_TTSTOP in tp->t_state.
+ */
+ int zst_tbc; /* transmit byte count */
+ caddr_t zst_tba; /* transmit buffer address */
+ int zst_heldtbc; /* held tbc while xmission stopped */
+
+ /* Flags to communicate with zstty_softint() */
+ volatile char zst_rx_blocked; /* input block at ring */
+ volatile char zst_rx_overrun; /* ring overrun */
+ volatile char zst_tx_busy; /* working on an output chunk */
+ volatile char zst_tx_done; /* done with one output chunk */
+ volatile char zst_tx_stopped; /* H/W level stop (lost CTS) */
+ volatile char zst_st_check; /* got a status interrupt */
+ char pad[2];
+};
/* Definition of the driver for autoconfig. */
@@ -104,21 +168,20 @@ static int zsparam(struct tty *, struct termios *);
static void zs_modem(struct zstty_softc *zst, int onoff);
static int zshwiflow(struct tty *, int);
static void zs_hwiflow(struct zstty_softc *, int);
-static int zsgetbaud(register struct zs_chanstate *,
- register int *rate, register int *tc, register u_char *rr4,
- register u_char *rr11, register u_char *rr14,
- register int *source, register int *sourceflag);
-
+static void zstty_rxint(register struct zs_chanstate *);
+static void zstty_txint(register struct zs_chanstate *);
+static void zstty_stint(register struct zs_chanstate *);
+static void zstty_softint(struct zs_chanstate *);
+static void zsoverrun(struct zstty_softc *, long *, char *);
/*
* zstty_match: how is this zs channel configured?
*/
int
-zstty_match(parent, vcf, aux)
+zstty_match(parent, match, aux)
struct device *parent;
- void *vcf;
- void *aux;
+ void *match, *aux;
{
- struct cfdata *cf = (struct cfdata *) vcf;
+ struct cfdata *cf = match;
struct zsc_attach_args *args = aux;
/* Exact match is better than wildcard. */
@@ -147,31 +210,22 @@ zstty_attach(parent, self, aux)
int channel, tty_unit;
dev_t dev;
- cf = zst->zst_dev.dv_cfdata;
tty_unit = zst->zst_dev.dv_unit;
channel = args->channel;
- cs = &zsc->zsc_cs[channel];
+ cs = zsc->zsc_cs[channel];
cs->cs_private = zst;
cs->cs_ops = &zsops_tty;
zst->zst_cs = cs;
zst->zst_swflags = cf->cf_flags; /* softcar, etc. */
zst->zst_hwflags = args->hwflags;
-
- zst->zst_cflag = ZSTTY_DEF_CFLAG; /* set up defaults */
- zst->zst_iflag = TTYDEF_IFLAG; /* an ioctl can change */
- zst->zst_lflag = TTYDEF_LFLAG; /* these values, modifying */
- zst->zst_oflag = TTYDEF_OFLAG; /* initial defaults */
- zst->zst_ispeed = zst->zst_ospeed = cs->cs_defspeed;
- /* zst_cc set after tty is malloc'd */
-
- dev = makedev(ZSTTY_MAJOR, tty_unit);
+ dev = makedev(zs_major, tty_unit);
if (zst->zst_swflags)
printf(" flags 0x%x", zst->zst_swflags);
if (zst->zst_hwflags & ZS_HWFLAG_CONSOLE)
- printf(" (console)");
+ printf(": console");
else {
#ifdef KGDB
/*
@@ -189,50 +243,39 @@ zstty_attach(parent, self, aux)
}
#endif
}
+ printf("\n");
tp = ttymalloc();
tp->t_dev = dev;
tp->t_oproc = zsstart;
tp->t_param = zsparam;
tp->t_hwiflow = zshwiflow;
- ttychars(tp);
- bcopy(tp->t_cc, zst->zst_cc, sizeof(tp->t_cc));
zst->zst_tty = tp;
zst->zst_rbhiwat = zstty_rbuf_size; /* impossible value */
zst->zst_ringmask = zstty_rbuf_size - 1;
zst->zst_rbuf = malloc(zstty_rbuf_size * sizeof(zst->zst_rbuf[0]),
M_DEVBUF, M_WAITOK);
+ /* XXX - Do we need an MD hook here? */
- zstty_mdattach(zsc, zst, cs, tp); /*let the md code customize stuff */
-
- if (zst->zst_hwflags & (ZS_HWFLAG_IGCTS | ZS_HWFLAG_IGDCD)) {
- printf("\n Ignoring ");
- switch (zst->zst_hwflags & (ZS_HWFLAG_IGCTS | ZS_HWFLAG_IGDCD)) {
- case ZS_HWFLAG_IGCTS:
- printf("CTS line "); break;
- case ZS_HWFLAG_IGDCD:
- printf("DCD line "); break;
- default:
- printf("CTS and DCD lines ");
- }
- }
-
- printf("\n");
/*
* Hardware init
*/
if (zst->zst_hwflags & ZS_HWFLAG_CONSOLE) {
- /* This unit is the console. */
+ /* Call zsparam similar to open. */
+ struct termios t;
+
+ /* Make console output work while closed. */
zst->zst_swflags |= TIOCFLAG_SOFTCAR;
- /* Call _param so interrupts get enabled. */
- bcopy(&zst->zst_termios, &tp->t_termios, sizeof(struct termios));
- /* copy the whole termios in as the first "first open" won't
- * do it since the speed != 0 */
- cs->cs_defspeed = zs_getspeed(cs);
- tp->t_ispeed = cs->cs_defspeed;
- tp->t_ospeed = cs->cs_defspeed;
- (void) zsparam(tp, &tp->t_termios);
+ /* Setup the "new" parameters in t. */
+ bzero((void*)&t, sizeof(t));
+ t.c_cflag = cs->cs_defcflag;
+ t.c_ospeed = cs->cs_defspeed;
+ /* Enable interrupts. */
+ cs->cs_preg[1] = ZSWR1_RIE | ZSWR1_SIE;
+ /* Make sure zsparam will see changes. */
+ tp->t_ospeed = 0;
+ (void) zsparam(tp, &t);
} else {
/* Not the console; may need reset. */
int reset, s;
@@ -296,10 +339,6 @@ zsopen(dev, flags, mode, p)
tp = zst->zst_tty;
cs = zst->zst_cs;
-#ifdef ZSTTYDEBUG
- zsprintf("zs_open to channel at %p\n",cs->cs_reg_csr);
-#endif
-
/* If KGDB took the line, then tp==NULL */
if (tp == NULL)
return (EBUSY);
@@ -315,57 +354,78 @@ zsopen(dev, flags, mode, p)
s = spltty();
if ((tp->t_state & TS_ISOPEN) == 0) {
- if ((tp->t_ispeed == 0) || (zst->zst_resetdef)) {
- /* First open. Executed if either the tty
- * was uninitialized, or if we choose to
- * reset defaults w/ each open. */
- bcopy(&zst->zst_termios, &tp->t_termios,
- sizeof(struct termios));
- if (zst->zst_swflags & TIOCFLAG_CLOCAL)
- tp->t_cflag |= CLOCAL;
- if (zst->zst_swflags & TIOCFLAG_CRTSCTS)
- tp->t_cflag |= CRTSCTS;
- if (zst->zst_swflags & TIOCFLAG_MDMBUF)
- tp->t_cflag |= MDMBUF;
+ /* First open. */
+ struct termios t;
+
+ /*
+ * Setup the "new" parameters in t.
+ * Can not use tp->t because zsparam
+ * deals only with what has changed.
+ */
+ bzero((void*)&t, sizeof(t));
+ t.c_cflag = cs->cs_defcflag;
+ if (zst->zst_swflags & TIOCFLAG_CLOCAL)
+ t.c_cflag |= CLOCAL;
+ if (zst->zst_swflags & TIOCFLAG_CRTSCTS)
+ t.c_cflag |= CRTSCTS;
+ if (zst->zst_swflags & TIOCFLAG_MDMBUF)
+ t.c_cflag |= MDMBUF;
+ t.c_ospeed = cs->cs_defspeed;
+ /* Enable interrupts. */
+ cs->cs_preg[1] = ZSWR1_RIE | ZSWR1_SIE;
+ /* Make sure zsparam will see changes. */
+ tp->t_ospeed = 0;
+ (void) zsparam(tp, &t);
+ /*
+ * Note: zsparam has done: cflag, ispeed, ospeed
+ * so we just need to do: iflag, oflag, lflag, cc
+ * For "raw" mode, just leave all zeros.
+ */
+ if ((zst->zst_hwflags & ZS_HWFLAG_RAW) == 0) {
+ tp->t_iflag = TTYDEF_IFLAG;
+ tp->t_oflag = TTYDEF_OFLAG;
+ tp->t_lflag = TTYDEF_LFLAG;
+ ttychars(tp);
}
- (void) zsparam(tp, &tp->t_termios);
ttsetwater(tp);
/* Flush any pending input. */
zst->zst_rbget = zst->zst_rbput;
zs_iflush(cs); /* XXX */
- /* Turn on DTR */
- zs_modem(zst, 1);
+ /* DTR was turned on by zsparam. */
if (zst->zst_swflags & TIOCFLAG_SOFTCAR) {
tp->t_state |= TS_CARR_ON;
}
+ /* XXX - The MD code could just force CLOCAL instead. */
+ if (zst->zst_hwflags & ZS_HWFLAG_NO_DCD) {
+ tp->t_state |= TS_CARR_ON;
+ }
}
error = 0;
- /* Wait for carrier. */
-#ifdef ZSTTYDEBUG
- zsprintf("wait for carrier...\n");
-#endif
- for (;;) {
+ /* In this section, we may touch the chip. */
+ (void)splzs();
- if (zst->zst_hwflags & ZS_HWFLAG_IGDCD) {
- tp->t_state |= TS_CARR_ON;
- break;
- }
+ /*
+ * Get initial value of RR0. This is done after we
+ * raise DTR in case the cable loops DTR back to CTS.
+ */
+ cs->cs_rr0 = zs_read_csr(cs);
- /* Might never get status intr if carrier already on. */
- cs->cs_rr0 = zs_read_csr(cs);
- if (cs->cs_rr0 & ZSRR0_DCD) {
+ /*
+ * Wait for DCD (if necessary). Note that we might
+ * never get status interrupt if DCD is already on.
+ */
+ for (;;) {
+ /* Check the DCD bit (if we have one). */
+ if (cs->cs_rr0 & cs->cs_rr0_dcd)
tp->t_state |= TS_CARR_ON;
- break;
- }
if ((tp->t_state & TS_CARR_ON) ||
(tp->t_cflag & CLOCAL) ||
(flags & O_NONBLOCK) )
- {
break;
- }
+ /* Sleep waiting for a status interrupt. */
tp->t_state |= TS_WOPEN;
error = ttysleep(tp, (caddr_t)&tp->t_rawq,
TTIPRI | PCATCH, ttopen, 0);
@@ -378,18 +438,12 @@ zsopen(dev, flags, mode, p)
}
break;
}
+ /* The status interrupt changed cs->cs_rr0 */
}
splx(s);
-
-#ifdef ZSTTYDEBUG
- zsprintf("...carrier %s\n",
- (tp->t_state & TS_CARR_ON) ? "on" : "off");
-#endif
-
if (error == 0)
error = linesw[tp->t_line].l_open(dev, tp);
-
return (error);
}
@@ -406,11 +460,7 @@ zsclose(dev, flags, mode, p)
struct zstty_softc *zst;
register struct zs_chanstate *cs;
register struct tty *tp;
- int hup;
-
-#ifdef ZSTTYDEBUG
- zsprintf("zs_close\n");
-#endif
+ int hup, s;
zst = zstty_cd.cd_devs[minor(dev)];
cs = zst->zst_cs;
@@ -421,6 +471,14 @@ zsclose(dev, flags, mode, p)
return 0;
(*linesw[tp->t_line].l_close)(tp, flags);
+
+ /* Disable interrupts. */
+ s = splzs();
+ cs->cs_creg[1] = cs->cs_preg[1] = 0;
+ zs_write_reg(cs, 1, cs->cs_creg[1]);
+ splx(s);
+
+ /* Maybe do "hangup" (drop DTR). */
hup = tp->t_cflag & HUPCL;
if (zst->zst_swflags & TIOCFLAG_SOFTCAR)
hup = 0;
@@ -432,7 +490,6 @@ zsclose(dev, flags, mode, p)
if (cs->cs_creg[5] & ZSWR5_BREAK) {
zs_break(cs, 0);
}
- /* XXX - turn off interrupts? */
ttyclose(tp);
return (0);
@@ -495,9 +552,12 @@ zsioctl(dev, cmd, data, flag, p)
error = ttioctl(tp, cmd, data, flag, p);
if (error >= 0)
return (error);
- error = zsmdioctl(tp, cmd, data, flag, p);
- if (error >= 0)
- return (error);
+
+ #ifdef ZS_MD_IOCTL
+ error = ZS_MD_IOCTL;
+ if (error >= 0)
+ return (error);
+ #endif /* ZS_MD_IOCTL */
switch (cmd) {
@@ -536,30 +596,6 @@ zsioctl(dev, cmd, data, flag, p)
zs_modem(zst, 0);
break;
-#if 0
- case SetSafeOpen:
- error = suser(p, 0);
- if (error != 0)
- return (EPERM);
- zst->zst_resetdef = 1;
- break;
-
- case ClearSafeOpen:
- error = suser(p, 0);
- if (error != 0)
- return (EPERM);
- zst->zst_resetdef = 0;
- break;
-
- case SetOpenDefaults:
- error = suser(p, 0);
- if (error != 0)
- return (EPERM);
- bcopy(&tp->t_termios, &zst->zst_termios, \
- sizeof(struct termios));
- break;
-#endif
-
case TIOCMSET:
case TIOCMBIS:
case TIOCMBIC:
@@ -594,10 +630,10 @@ zsstart(tp)
/*
* If under CRTSCTS hfc and halted, do nothing
+ * This flag can only be set with CRTSCTS.
*/
- if (tp->t_cflag & CRTSCTS)
- if (zst->zst_tx_stopped)
- goto out;
+ if (zst->zst_tx_stopped)
+ goto out;
/*
* If there are sleepers, and output has drained below low
@@ -611,15 +647,15 @@ zsstart(tp)
selwakeup(&tp->t_wsel);
}
+ nch = ndqb(&tp->t_outq, 0); /* XXX */
(void) splzs();
- nch = ndqb(&tp->t_outq, 0); /* XXX */
if (nch) {
register char *p = tp->t_outq.c_cf;
/* mark busy, enable tx done interrupts, & send first byte */
tp->t_state |= TS_BUSY;
-
+ zst->zst_tx_busy = 1;
cs->cs_preg[1] |= ZSWR1_TIE;
cs->cs_creg[1] = cs->cs_preg[1];
zs_write_reg(cs, 1, cs->cs_creg[1]);
@@ -658,144 +694,15 @@ zsstop(tp, flag)
if (tp->t_state & TS_BUSY) {
/*
* Device is transmitting; must stop it.
+ * Also clear _heldtbc to prevent any
+ * flow-control event from resuming.
*/
zst->zst_tbc = 0;
- zst->zst_heldtbc = 0; /* XXX */
+ zst->zst_heldtbc = 0;
if ((tp->t_state & TS_TTSTOP) == 0)
tp->t_state |= TS_FLUSH;
}
splx(s);
- return(0);
-}
-
-#ifndef ZS_TOLERANCE
-#define ZS_TOLERANCE 50
-/* 5% in tenths of a % */
-#endif
-
-/*
- * Search through the signal sources in the channel, and
- * pick the best one for the baud rate requested. Return
- * a -1 if not achievable in tolerance. Otherwise ret 0
- * and fill in the values.
- *
- * This routine draws inspiration from the Atari port's zs.c
- * driver in NetBSD1.1 which did the same type of source switching.
- * Tolerance code inspired by comspeed routine in isa/com.c.
- *
- * By Bill Studenmund, 1996-05-12
- */
-static int
-zsgetbaud(cs, rate, tc, rr4, rr11, rr14, source, sourceflag)
- register struct zs_chanstate *cs;
- register int *rate, *tc, *source, *sourceflag;
- register u_char *rr4, *rr11, *rr14;
-{
- int i, tc0, tc1, s, sf, rate0, rate1, err, tol;
-
- s = -1; /* no valid source yet */
- tol = ZS_TOLERANCE;
-
- sf = 0; tc0 = 0; rate0 = 0; /* XXX Kill gcc warning */
-
- /*
- * Step through all the sources and see which one matches
- * the best. A source has to match BETTER than tol to be chosen.
- * Thus if two sources give the same error, the first one will be
- * chosen. Also, allow for the possability that one source might run
- * both the BRG and the direct divider (i.e. RTxC).
- */
- for (i=0; i< cs->cs_clock_count; i++) {
- if (cs->cs_clocks[i].clk <= 0)
- continue; /* skip non-existent or bad clocks */
- if (cs->cs_clocks[i].flags & ZSC_BRG) {
- /* check out BRG at /16 */
- tc1 = BPS_TO_TCONST(cs->cs_clocks[i].clk >> 4, rate[0]);
- if (tc1 >= 0) {
- rate1 = TCONST_TO_BPS(cs->cs_clocks[i].clk >> 4, tc1);
- err = abs(((rate1 - rate[0])*1000)/rate[0]);
- if (err < tol) {
- tol = err;
- s = i;
- sf = cs->cs_clocks[i].flags & ~ZSC_DIV;
- tc0 = tc1;
- rate0 = rate1;
- }
- }
- }
- if (cs->cs_clocks[i].flags & ZSC_DIV) {
- /*
- * Check out either /1, /16, /32, or /64
- * Note: for /1, you'd better be using a synchronized
- * clock!
- */
- int b0 = cs->cs_clocks[i].clk, e0 = abs(b0-rate[0]);
- int b1 = b0 >> 4, e1 = abs(b1-rate[0]);
- int b2 = b1 >> 1, e2 = abs(b2-rate[0]);
- int b3 = b2 >> 1, e3 = abs(b3-rate[0]);
-
- if (e0 < e1 && e0 < e2 && e0 < e3) {
- err = e0;
- rate1 = b0;
- tc1 = ZSWR4_CLK_X1;
- } else if (e0 > e1 && e1 < e2 && e1 < e3) {
- err = e1;
- rate1 = b1;
- tc1 = ZSWR4_CLK_X16;
- } else if (e0 > e2 && e1 > e2 && e2 < e3) {
- err = e2;
- rate1 = b2;
- tc1 = ZSWR4_CLK_X32;
- } else {
- err = e3;
- rate1 = b3;
- tc1 = ZSWR4_CLK_X64;
- }
-
- err = (err * 1000)/rate[0];
- if (err < tol) {
- tol = err;
- s = i;
- sf = cs->cs_clocks[i].flags & ~ZSC_BRG;
- tc0 = tc1;
- rate0 = rate1;
- }
- }
- }
-#ifdef ZSTTYDEBUG
- zsprintf("Checking for rate %d. Found source #%d.\n",rate[0], s);
-#endif
- if (s == -1) return (-1);
- /*
- * Now we have a source, so set it up.
- */
- *source = s;
- *sourceflag = sf;
- rate[0] = rate0;
- if (sf & ZSC_BRG) {
- *rr4 = ZSWR4_CLK_X16;
- *rr11= ZSWR11_RXCLK_BAUD | ZSWR11_TXCLK_BAUD;
- if (sf & ZSC_PCLK) {
- *rr14 = ZSWR14_BAUD_ENA | ZSWR14_BAUD_FROM_PCLK;
- } else {
- *rr14 = ZSWR14_BAUD_ENA;
- }
- *tc = tc0;
- } else {
- *rr4 = tc0;
- if (sf & ZSC_RTXDIV) {
- *rr11 = ZSWR11_RXCLK_RTXC | ZSWR11_TXCLK_RTXC;
- } else {
- *rr11 = ZSWR11_RXCLK_TRXC | ZSWR11_TXCLK_TRXC;
- }
- *rr14= 0;
- *tc = 0xffff;
- }
-#ifdef ZSTTYDEBUG
- zsprintf("Rate is %7d, tc is %7d, source no. %2d, flags %4x\n", \
- *rate, *tc, *source, *sourceflag);
- zsprintf("Registers are: 4 %x, 11 %x, 14 %x\n\n", *rr4, *rr11, *rr14);
-#endif
return (0);
}
@@ -803,154 +710,146 @@ zsgetbaud(cs, rate, tc, rr4, rr11, rr14, source, sourceflag)
* Set ZS tty parameters from termios.
* XXX - Should just copy the whole termios after
* making sure all the changes could be done.
- * XXX - Only whack the UART when params change...
*/
static int
zsparam(tp, t)
register struct tty *tp;
register struct termios *t;
{
- register struct zstty_softc *zst;
- register struct zs_chanstate *cs;
- register int s, cflag, hfc;
- u_char tmp3, tmp4, tmp5, tmp11, tmp14;
- int bps, tconst, src, srcflag;
+ struct zstty_softc *zst;
+ struct zs_chanstate *cs;
+ int s, bps, cflag, error;
+ u_char tmp3, tmp4, tmp5;
zst = zstty_cd.cd_devs[minor(tp->t_dev)];
cs = zst->zst_cs;
-
bps = t->c_ospeed;
+ cflag = t->c_cflag;
+
if (bps < 0 || (t->c_ispeed && t->c_ispeed != bps))
return (EINVAL);
- if (bps == 0) {
- /* stty 0 => drop DTR and RTS */
- zs_modem(zst, 0);
+
+ /*
+ * Only whack the UART when params change.
+ * Some callers need to clear tp->t_ospee
+ * to make sure initialization gets done.
+ */
+ if ((tp->t_ospeed == bps) &&
+ (tp->t_cflag == cflag) )
return (0);
- }
- if (0 > zsgetbaud(cs, &bps, &tconst, &tmp4, &tmp11, &tmp14,
- &src, &srcflag))
- return (EINVAL);
-
- tp->t_ispeed = tp->t_ospeed = bps;
- cs->cs_psource = src;
- cs->cs_pclk_flag = srcflag;
-
- cflag = t->c_cflag;
/*
- * Make sure we don't enable hfc on a signal line we're ignoring
- *
- * As we enable CTS interrupts only if we have CRTSCTS, this code
- * also effectivly turns off ZSWR15_CTS_IE.
+ * Call MD functions to deal with changed
+ * clock modes or H/W flow control modes.
+ * The BRG divisor is set now. (reg 12,13
*/
- if (zst->zst_hwflags & ZS_HWFLAG_IGDCD)
- cflag &= ~MDMBUF;
- if (zst->zst_hwflags & ZS_HWFLAG_IGCTS)
- cflag &= ~CRTSCTS;
+ error = zs_set_speed(cs, bps);
+ if (error)
+ return (error);
+ error = zs_set_modes(cs, cflag);
+ if (error)
+ return (error);
+
+ /* OK, we are now committed to do it. */
tp->t_cflag = cflag;
+ tp->t_ospeed = bps;
+ tp->t_ispeed = bps;
+
/*
* Block interrupts so that state will not
* be altered until we are done setting it up.
- */
- s = splzs();
-
- /*
+ *
* Initial values in cs_preg are set before
* our attach routine is called. The master
* interrupt enable is handled by zsc.c
*/
+ s = splzs();
- cs->cs_preg[12] = tconst & 255;
- cs->cs_preg[13] = tconst >> 8;
-
+ /* Recompute character size bits. */
+ tmp3 = cs->cs_preg[3] & ~ZSWR3_RXSIZE;
+ tmp5 = cs->cs_preg[5] & ~ZSWR5_TXSIZE;
switch (cflag & CSIZE) {
case CS5:
- tmp3 = ZSWR3_RX_5;
- tmp5 = ZSWR5_TX_5;
+ /* These are |= 0 but let the optimizer deal with it. */
+ tmp3 |= ZSWR3_RX_5;
+ tmp5 |= ZSWR5_TX_5;
break;
case CS6:
- tmp3 = ZSWR3_RX_6;
- tmp5 = ZSWR5_TX_6;
+ tmp3 |= ZSWR3_RX_6;
+ tmp5 |= ZSWR5_TX_6;
break;
case CS7:
- tmp3 = ZSWR3_RX_7;
- tmp5 = ZSWR5_TX_7;
+ tmp3 |= ZSWR3_RX_7;
+ tmp5 |= ZSWR5_TX_7;
break;
case CS8:
default:
- tmp3 = ZSWR3_RX_8;
- tmp5 = ZSWR5_TX_8;
+ tmp3 |= ZSWR3_RX_8;
+ tmp5 |= ZSWR5_TX_8;
break;
}
+ /* Raise or lower DTR and RTS as appropriate. */
+ if (bps) {
+ /* Raise DTR and RTS */
+ tmp5 |= cs->cs_wr5_dtr;
+ } else {
+ /* Drop DTR and RTS */
+ /* XXX: Should SOFTCAR prevent this? */
+ tmp5 &= ~(cs->cs_wr5_dtr);
+ }
+ cs->cs_preg[3] = tmp3;
+ cs->cs_preg[5] = tmp5;
- cs->cs_preg[3] = tmp3 | ZSWR3_RX_ENABLE;
- cs->cs_preg[5] = tmp5 | ZSWR5_TX_ENABLE | ZSWR5_DTR | ZSWR5_RTS;
-
- tmp4 |= (cflag & CSTOPB ? ZSWR4_TWOSB : ZSWR4_ONESB);
+ /*
+ * Recompute the stop bits and parity bits. Note that
+ * zs_set_speed() may have set clock selection bits etc.
+ * in wr4, so those must preserved.
+ */
+ tmp4 = cs->cs_preg[4];
+ /* Recompute stop bits. */
+ tmp4 &= ~ZSWR4_SBMASK;
+ tmp4 |= (cflag & CSTOPB) ?
+ ZSWR4_TWOSB : ZSWR4_ONESB;
+ /* Recompute parity bits. */
+ tmp4 &= ~ZSWR4_PARMASK;
if ((cflag & PARODD) == 0)
tmp4 |= ZSWR4_EVENP;
if (cflag & PARENB)
tmp4 |= ZSWR4_PARENB;
cs->cs_preg[4] = tmp4;
- /*
- * Output hardware flow control on the chip is horrendous:
- * if carrier detect drops, the receiver is disabled and if
- * CTS drops, the transmitter is stoped IN MID CHARACTER!
- * Therefore, do not set the HFC bit, and instead use
- * the status interrupts to detect CTS changes.
- */
- if (cflag & CRTSCTS) {
- zst->zst_rbhiwat = zstty_rbuf_hiwat;
- cs->cs_preg[15] |= ZSWR15_CTS_IE;
- } else {
- zst->zst_rbhiwat = zstty_rbuf_size; /* impossible value */
- cs->cs_preg[15] &= ~ZSWR15_CTS_IE;
- }
- /* Shouldn't the zst_rbhiwat stuff be if (cflag & CHWFLOW) ? wrs */
-
- /*
- * Disable DCD interrupts if we've been told to ignore
- * the DCD pin. Happens on mac68k because the input line for
- * DCD can also be used as a clock input.
- */
- if (zst->zst_hwflags & ZS_HWFLAG_IGDCD)
- cs->cs_preg[15] &= ~ZSWR15_DCD_IE;
- else
- cs->cs_preg[15] |= ZSWR15_DCD_IE;
-
- /*
- * now find out which line to change for input flow control.
- * Important as some ports (mac68k) don't always have input
- * flow control when they have output flow control (RTS actually
- * controls buffers on the Xmitter output).
- */
- hfc = (cflag & CRTSCTS) ? 1 : 0;
- hfc |= (cflag & MDMBUF) ? 2 : 0;
-
- zst->zst_hwimask = zst->zst_hwimasks[hfc];
- if ((zst->zst_rx_blocked) && (zst->zst_hwimask))
- cs->cs_preg[5] &= ~zst->zst_hwimask;
- /* make sure we enforce halted-ness */
-
- /* XXX check for loss of output blocking if loosing hwi ability? */
+ /* The MD function zs_set_modes handled CRTSCTS, etc. */
/*
* If nothing is being transmitted, set up new current values,
* else mark them as pending.
*/
if (cs->cs_heldchange == 0) {
- if (tp->t_state & TS_BUSY) {
+ if (zst->zst_tx_busy) {
zst->zst_heldtbc = zst->zst_tbc;
zst->zst_tbc = 0;
- cs->cs_heldchange = 0xFF;
+ cs->cs_heldchange = 0xFFFF;
} else {
zs_loadchannelregs(cs);
}
}
splx(s);
- /* check for ttstart if lost output flow control? XXX */
+
+ /* If we can throttle input, enable "high water" detection. */
+ if (cflag & CHWFLOW) {
+ zst->zst_rbhiwat = zstty_rbuf_hiwat;
+ } else {
+ /* This impossible value prevents a "high water" trigger. */
+ zst->zst_rbhiwat = zstty_rbuf_size;
+ /* XXX: Lost hwi ability, so unblock and restart. */
+ zst->zst_rx_blocked = 0;
+ if (zst->zst_tx_stopped) {
+ zst->zst_tx_stopped = 0;
+ zsstart(tp);
+ }
+ }
+
return (0);
}
@@ -964,23 +863,25 @@ zs_modem(zst, onoff)
int onoff;
{
struct zs_chanstate *cs;
- struct tty *tp;
- int s, bis, and;
+ int s, clr, set;
cs = zst->zst_cs;
- tp = zst->zst_tty;
+ if (cs->cs_wr5_dtr == 0)
+ return;
if (onoff) {
- bis = ZSWR5_DTR | ZSWR5_RTS;
- and = ~0;
+ clr = 0;
+ set = cs->cs_wr5_dtr;
} else {
- bis = 0;
- and = ~(ZSWR5_DTR | ZSWR5_RTS);
+ clr = cs->cs_wr5_dtr;
+ set = 0;
}
+
s = splzs();
- cs->cs_preg[5] = (cs->cs_preg[5] | bis) & and;
+ cs->cs_preg[5] &= ~clr;
+ cs->cs_preg[5] |= set;
if (cs->cs_heldchange == 0) {
- if (tp->t_state & TS_BUSY) {
+ if (zst->zst_tx_busy) {
zst->zst_heldtbc = zst->zst_tbc;
zst->zst_tbc = 0;
cs->cs_heldchange = (1<<5);
@@ -1004,20 +905,16 @@ zshwiflow(tp, stop)
int stop;
{
register struct zstty_softc *zst;
+ register struct zs_chanstate *cs;
int s;
zst = zstty_cd.cd_devs[minor(tp->t_dev)];
+ cs = zst->zst_cs;
+
+ /* Can not do this without some bit assigned as RTS. */
+ if (cs->cs_wr5_rts == 0)
+ return (0);
- /*
- * This loop checks to see that we can in fact control input.
- * If not, then do little except tell the upper layer the truth.
- */
- if (zst->zst_hwimask == 0) {
- if (stop)
- return 0;
- else
- return 1; /* yes, w/o hwi we can unblock input. ;-) */
- }
s = splzs();
if (stop) {
/*
@@ -1027,14 +924,12 @@ zshwiflow(tp, stop)
if (zst->zst_rx_blocked)
goto out;
zst->zst_rx_blocked = 1;
- z8530tty_stats.tty_block++;
} else {
/*
* The tty layer is asking us to resume input.
* The input ring is always empty by now.
*/
zst->zst_rx_blocked = 0;
- z8530tty_stats.tty_unblock++;
}
zs_hwiflow(zst, stop);
out:
@@ -1052,25 +947,27 @@ zs_hwiflow(zst, stop)
int stop;
{
register struct zs_chanstate *cs;
- register struct tty *tp;
- register int bis, and;
+ register int clr, set;
cs = zst->zst_cs;
- tp = zst->zst_tty;
+
+ if (cs->cs_wr5_rts == 0)
+ return;
if (stop) {
/* Block input (Lower RTS) */
- bis = 0;
- and = ~zst->zst_hwimask;
+ clr = cs->cs_wr5_rts;
+ set = 0;
} else {
/* Unblock input (Raise RTS) */
- bis = zst->zst_hwimask;
- and = ~0;
+ clr = 0;
+ set = cs->cs_wr5_rts;
}
- cs->cs_preg[5] = (cs->cs_preg[5] | bis) & and;
+ cs->cs_preg[5] &= ~clr;
+ cs->cs_preg[5] |= set;
if (cs->cs_heldchange == 0) {
- if (tp->t_state & TS_BUSY) {
+ if (zst->zst_tx_busy) {
zst->zst_heldtbc = zst->zst_tbc;
zst->zst_tbc = 0;
cs->cs_heldchange = (1<<5);
@@ -1086,14 +983,9 @@ zs_hwiflow(zst, stop)
* Interface to the lower layer (zscc)
****************************************************************/
-static void zstty_rxint(register struct zs_chanstate *);
-static void zstty_txint(register struct zs_chanstate *);
-static void zstty_stint(register struct zs_chanstate *);
-static void zsoverrun(struct zstty_softc *, long *, char *);
-static void zstty_softint(struct zs_chanstate *);
-/*
- * XXX: need to do input flow-control to avoid ring overrun.
- */
+static void zstty_rxint (struct zs_chanstate *);
+static void zstty_txint (struct zs_chanstate *);
+static void zstty_stint (struct zs_chanstate *);
/*
* receiver ready interrupt.
@@ -1157,7 +1049,6 @@ nextchar:
if ((cc > zst->zst_rbhiwat) && (zst->zst_rx_blocked == 0)) {
zst->zst_rx_blocked = 1;
zs_hwiflow(zst, 1);
- z8530tty_stats.ring_block++;
}
/* Ask for softint() call. */
@@ -1211,7 +1102,8 @@ zstty_txint(cs)
zs_write_csr(cs, ZSWR0_RESET_TXINT);
/* Ask the softint routine for more output. */
- zst->zst_tx_empty = 1;
+ zst->zst_tx_busy = 0;
+ zst->zst_tx_done = 1;
cs->cs_softreq = 1;
}
@@ -1223,11 +1115,9 @@ zstty_stint(cs)
register struct zs_chanstate *cs;
{
register struct zstty_softc *zst;
- register struct tty *tp;
- register u_char rr0;
+ register u_char rr0, delta;
zst = cs->cs_private;
- tp = zst->zst_tty;
rr0 = zs_read_csr(cs);
zs_write_csr(cs, ZSWR0_RESET_STATUS);
@@ -1236,27 +1126,38 @@ zstty_stint(cs)
* Check here for console break, so that we can abort
* even when interrupts are locking up the machine.
*/
- if ((rr0 & ZSRR0_BREAK))
+ if ((rr0 & ZSRR0_BREAK) &&
+ (zst->zst_hwflags & ZS_HWFLAG_CONSOLE))
{
- zs_abort(zst);
+ zs_abort(cs);
return;
}
/*
+ * We have to accumulate status line changes here.
+ * Otherwise, if we get multiple status interrupts
+ * before the softint runs, we could fail to notice
+ * some status line changes in the softint routine.
+ * Fix from Bill Studenmund, October 1996.
+ */
+ delta = (cs->cs_rr0 ^ rr0);
+ cs->cs_rr0_delta |= delta;
+ cs->cs_rr0 = rr0;
+
+ /*
* Need to handle CTS output flow control here.
* Output remains stopped as long as either the
* zst_tx_stopped or TS_TTSTOP flag is set.
* Never restart here; the softint routine will
* do that after things are ready to move.
*/
- if (((rr0 & ZSRR0_CTS) == 0) && (tp->t_cflag & CRTSCTS)) {
+ if ((delta & cs->cs_rr0_cts) &&
+ ((rr0 & cs->cs_rr0_cts) == 0))
+ {
zst->zst_tbc = 0;
zst->zst_heldtbc = 0;
zst->zst_tx_stopped = 1;
}
-
- cs->cs_rr0_changes |= cs->cs_rr0 ^ rr0;
- cs->cs_rr0 = rr0;
zst->zst_st_check = 1;
/* Ask for softint() call. */
@@ -1273,8 +1174,8 @@ zsoverrun(zst, ptime, what)
char *what;
{
- if (*ptime != time.tv_sec) {
- *ptime = time.tv_sec;
+ if (*ptime != time_second) {
+ *ptime = time_second;
log(LOG_WARNING, "%s: %s overrun\n",
zst->zst_dev.dv_xname, what);
}
@@ -1302,7 +1203,7 @@ zstty_softint(cs)
register int get, c, s;
int ringmask, overrun;
register u_short ring_data;
- register u_char rr0, delta, flag;
+ register u_char rr0, delta;
zst = cs->cs_private;
tp = zst->zst_tty;
@@ -1319,12 +1220,6 @@ zstty_softint(cs)
zst->zst_rx_overrun = 0;
zsoverrun(zst, &zst->zst_rotime, "ring");
}
- if (cs->cs_flags & ZS_FLAGS_INTERRUPT_OVERRUN) {
- (void) splzs();
- cs->cs_flags &= ~ZS_FLAGS_INTERRUPT_OVERRUN;
- (void) spltty();
- zsoverrun(zst, &zst->zst_intotime, "interrupt");
- }
/*
* Copy data from the receive ring into the tty layer.
@@ -1366,7 +1261,6 @@ zstty_softint(cs)
(void) splzs();
zst->zst_rx_blocked = 0;
zs_hwiflow(zst, 0); /* unblock input */
- z8530tty_stats.ring_unblock++;
(void) spltty();
}
@@ -1378,50 +1272,41 @@ zstty_softint(cs)
if (zst->zst_st_check) {
zst->zst_st_check = 0;
+ (void) splzs();
rr0 = cs->cs_rr0;
- delta = cs->cs_rr0_changes;
- cs->cs_rr0_changes = 0;
- if ((delta & ZSRR0_DCD) &&
- ~(zst->zst_hwflags & ZS_HWFLAG_IGDCD)) {
- c = ((rr0 & ZSRR0_DCD) != 0);
- if ((tp->t_cflag & CHWFLOW) == CHWFLOW) {
- flag = 1;
- tp->t_cflag &= ~MDMBUF;
- } else
- flag = 0;
+ delta = cs->cs_rr0_delta;
+ cs->cs_rr0_delta = 0;
+ (void) spltty();
+
+ /* Note, the MD code may use DCD for something else. */
+ if (delta & cs->cs_rr0_dcd) {
+ c = ((rr0 & cs->cs_rr0_dcd) != 0);
if (line->l_modem(tp, c) == 0)
zs_modem(zst, c);
- if (flag)
- tp->t_cflag |= MDMBUF;
- /*
- * The above trick hides MDMBUF from the tty layer
- * if we also have CRTSCTS; Used as mac68k takes
- * the two of them as meaning do CRTSCTS with DCD
- * as the hwi line. Just CRTSCTS doesn't have a
- * hwi line.
- */
}
- if ((delta & ZSRR0_CTS) && (tp->t_cflag & CRTSCTS)) {
+
+ /* Note, cs_rr0_cts is set only with H/W flow control. */
+ if (delta & cs->cs_rr0_cts) {
/*
* Only do restart here. Stop is handled
* at the h/w interrupt level.
*/
- if (rr0 & ZSRR0_CTS) {
+ if (rr0 & cs->cs_rr0_cts) {
zst->zst_tx_stopped = 0;
- tp->t_state &= ~TS_TTSTOP;
+ /* tp->t_state &= ~TS_TTSTOP; */
(*line->l_start)(tp);
}
}
}
- if (zst->zst_tx_empty) {
- zst->zst_tx_empty = 0;
+ if (zst->zst_tx_done) {
+ zst->zst_tx_done = 0;
tp->t_state &= ~TS_BUSY;
if (tp->t_state & TS_FLUSH)
tp->t_state &= ~TS_FLUSH;
else
ndflush(&tp->t_outq, zst->zst_tba -
- (caddr_t) tp->t_outq.c_cf);
+ (caddr_t) tp->t_outq.c_cf);
line->l_start(tp);
}
@@ -1435,4 +1320,3 @@ struct zsops zsops_tty = {
zstty_softint, /* process software interrupt */
};
-
diff --git a/sys/arch/mac68k/dev/z8530tty.h b/sys/arch/mac68k/dev/z8530tty.h
deleted file mode 100644
index ba602c6c178..00000000000
--- a/sys/arch/mac68k/dev/z8530tty.h
+++ /dev/null
@@ -1,140 +0,0 @@
-/* $OpenBSD: z8530tty.h,v 1.3 2003/06/02 23:27:49 millert Exp $ */
-/* $NetBSD: z8530tty.h,v 1.1 1996/05/18 18:54:35 briggs Exp $ */
-
-/*
- * Copyright (c) 1994 Gordon W. Ross
- * Copyright (c) 1992, 1993
- * The Regents of the University of California. All rights reserved.
- *
- * This software was developed by the Computer Systems Engineering group
- * at Lawrence Berkeley Laboratory under DARPA contract BG 91-66 and
- * contributed to Berkeley.
- *
- * All advertising materials mentioning features or use of this software
- * must display the following acknowledgement:
- * This product includes software developed by the University of
- * California, Lawrence Berkeley Laboratory.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- * 3. Neither the name of the University nor the names of its contributors
- * may be used to endorse or promote products derived from this software
- * without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
- * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
- * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
- * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
- * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
- * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
- * SUCH DAMAGE.
- *
- * @(#)zs.c 8.1 (Berkeley) 7/19/93
- */
-
-/*
- * Zilog Z8530 Dual UART driver (tty interface)
- *
- * This is the "slave" driver that will be attached to
- * the "zsc" driver for plain "tty" async. serial lines.
- */
-
-
-/*
- * Allow the MD var.h to override the default CFLAG so that
- * console messages during boot come out with correct parity.
- */
-#ifndef ZSTTY_DEF_CFLAG
-#define ZSTTY_DEF_CFLAG TTYDEF_CFLAG
-#endif
-
-/*
- * How many input characters we can buffer.
- * The port-specific var.h may override this.
- * Note: must be a power of two!
- */
-#ifndef ZSTTY_RING_SIZE
-#define ZSTTY_RING_SIZE 2048
-#endif
-
-struct zstty_stats {
- int ring_block;
- int ring_unblock;
- int tty_block;
- int tty_unblock;
-};
-
-struct zstty_softc {
- struct device zst_dev; /* required first: base device */
- struct tty *zst_tty;
- struct zs_chanstate *zst_cs;
-
- int zst_hwflags; /* see z8530var.h */
- int zst_swflags; /* TIOCFLAG_SOFTCAR, ... <ttycom.h> */
-
- /*
- * Printing an overrun error message often takes long enough to
- * cause another overrun, so we only print one per second.
- */
- long zst_rotime; /* time of last ring overrun */
- long zst_fotime; /* time of last fifo overrun */
- long zst_intotime; /* time of last interrupt overrun */
-
- /*
- * The receive ring buffer.
- */
- int zst_rbget; /* ring buffer `get' index */
- volatile int zst_rbput; /* ring buffer `put' index */
- int zst_ringmask;
- int zst_rbhiwat;
-
- u_short *zst_rbuf; /* rr1, data pairs */
-
- /*
- * The transmit byte count and address are used for pseudo-DMA
- * output in the hardware interrupt code. PDMA can be suspended
- * to get pending changes done; heldtbc is used for this. It can
- * also be stopped for ^S; this sets TS_TTSTOP in tp->t_state.
- */
- int zst_tbc; /* transmit byte count */
- caddr_t zst_tba; /* transmit buffer address */
- int zst_heldtbc; /* held tbc while xmission stopped */
-
- /* Flags to communicate with zstty_softint() */
- volatile char zst_rx_blocked;
- volatile char zst_rx_overrun;
- volatile char zst_tx_stopped;
- volatile char zst_tx_empty;
- volatile char zst_st_check;
- char pad[3];
-
- struct termios zst_termios; /* default values for tty flags */
-#define zst_cflag zst_termios.c_cflag
-#define zst_iflag zst_termios.c_iflag
-#define zst_lflag zst_termios.c_lflag
-#define zst_oflag zst_termios.c_oflag
-#define zst_cc zst_termios.c_cc
-#define zst_ispeed zst_termios.c_ispeed
-#define zst_ospeed zst_termios.c_ospeed
-
- char zst_resetdef; /* !=0 means reset tty defs. on open */
- char zst_hwimask; /* bits to keep low for hwiflow */
-
- char zst_hwimasks[4]; /* masks for hwiflow for HFC modes */
-};
-
-#define ZSTTY_RAW_CFLAG (CS8 | CREAD | HUPCL )
-#define ZSTTY_RAW_IFLAG (IXANY | IMAXBEL)
-#define ZSTTY_RAW_LFLAG (ECHOE|ECHOKE|ECHOCTL)
-#define ZSTTY_RAW_OFLAG (ONLCR | OXTABS)
-/* Above taken from looking at a tty after a stty raw */
diff --git a/sys/arch/mac68k/dev/zs.c b/sys/arch/mac68k/dev/zs.c
index ab82e6d875c..e8f6f83416b 100644
--- a/sys/arch/mac68k/dev/zs.c
+++ b/sys/arch/mac68k/dev/zs.c
@@ -1,7 +1,8 @@
-/* $OpenBSD: zs.c,v 1.15 2004/03/09 21:46:47 xsa Exp $ */
-/* $NetBSD: zs.c,v 1.12 1996/12/18 05:04:22 scottr Exp $ */
+/* $OpenBSD: zs.c,v 1.16 2004/11/25 18:32:10 miod Exp $ */
+/* $NetBSD: zs.c,v 1.19 1998/01/12 19:22:18 thorpej Exp $ */
/*
+ * Copyright (c) 1996-1998 Bill Studenmund
* Copyright (c) 1995 Gordon W. Ross
* All rights reserved.
*
@@ -37,6 +38,19 @@
* Runs two serial lines per chip using slave drivers.
* Plain tty/async lines use the zs_async slave.
* Sun keyboard/mouse uses the zs_kbd/zs_ms slaves.
+ * Other ports use their own mice & keyboard slaves.
+ *
+ * Credits & history:
+ *
+ * With NetBSD 1.1, port-mac68k started using a port of the port-sparc
+ * (port-sun3?) zs.c driver (which was in turn based on code in the
+ * Berkeley 4.4 Lite release). Bill Studenmund did the port, with
+ * help from Allen Briggs and Gordon Ross <gwr@netbsd.org>. Noud de
+ * Brouwer field-tested the driver at a local ISP.
+ *
+ * Bill Studenmund and Gordon Ross then ported the machine-independant
+ * z8530 driver to work with port-mac68k. NetBSD 1.2 contained an
+ * intermediate version (mac68k using a local, patched version of
*/
#include <sys/param.h>
@@ -52,47 +66,62 @@
#include <sys/syslog.h>
#include <dev/cons.h>
-#include "z8530reg.h"
+
+#include <mac68k/dev/z8530reg.h>
#include <machine/z8530var.h>
#include <machine/autoconf.h>
#include <machine/cpu.h>
+#include <machine/psc.h>
#include <machine/viareg.h>
+#ifdef DDB
#include <ddb/db_var.h>
+#endif
+
+/* Are these in a header file anywhere? */
+/* Booter flags interface */
+#define ZSMAC_RAW 0x01
+#define ZSMAC_LOCALTALK 0x02
+#define ZS_STD_BRG (57600*4)
+
+#include "zsc.h" /* get the # of zs chips defined */
/*
- * XXX: Hard code this to make console init easier...
+ * Some warts needed by z8530tty.c -
*/
-#define NZSC 1 /* XXX */
+int zs_def_cflag = (CREAD | CS8 | HUPCL);
+int zs_major = 12;
+
+/*
+ * abort detection on console will now timeout after iterating on a loop
+ * the following # of times. Cheep hack. Also, abort detection is turned
+ * off after a timeout (i.e. maybe there's not a terminal hooked up).
+ */
+#define ZSABORT_DELAY 3000000
/*
* Define interrupt levels.
*/
-#define ZSHARD_PRI 6 /* Wired on the CPU board... */
-#define ZSSOFT_PRI 3 /* Want tty pri (4) but this is OK. */
+#define ZSHARD_PRI 4 /* Wired on the CPU board... */
+/*
+ * Serial port cards with zs chips on them are actually at the
+ * NuBus interrupt level, which is lower than 4. But blocking
+ * level 4 interrupts will block those interrupts too, so level
+ * 4 is fine.
+ */
/* The layout of this is hardware-dependent (padding, order). */
struct zschan {
volatile u_char zc_csr; /* ctrl,status, and indirect access */
u_char zc_xxx0;
- u_char zc_xxx1;
- u_char zc_xxx2;
+ u_char zc_xxx1; /* part of the other channel lives here!*/
+ u_char zc_xxx2; /* Yea Apple! */
volatile u_char zc_data; /* data */
u_char zc_xxx3;
u_char zc_xxx4;
u_char zc_xxx5;
};
-/*
- * The zsdevice structure is not used on the mac68k port as the
- * chip is wired up weird. Channel B & A are interspersed with
- * the data & control bytes
-struct zsdevice {
- /! Yes, they are backwards. !/
- struct zschan zs_chan_b;
- struct zschan zs_chan_a;
-};
-*/
/* Saved PROM mappings */
static char *zsaddr[NZSC]; /* See zs_init() */
@@ -103,16 +132,19 @@ static int zs_defspeed[NZSC][2] = {
{ 9600, /* tty00 */
9600 }, /* tty01 */
};
-/* console stuff */
-void *zs_conschan = 0;
-int zs_consunit;
-/* device that the console is attached to--if serial. */
+void *zs_conschan = 0;
+int zs_consunit;
+/* device to which the console is attached--if serial. */
dev_t mac68k_zsdev;
-/* Mac stuff, some vestages of old mac serial driver here */
+/* Mac stuff */
volatile unsigned char *sccA = 0;
+int nzsc_attached = 0; /* needed as long as we have spurious
+ * interupt problems.
+ */
static struct zschan *zs_get_chan_addr(int zsc_unit, int channel);
void zs_init(void);
+int zs_cn_check_speed(int bps);
static struct zschan *
zs_get_chan_addr(zsc_unit, channel)
@@ -137,7 +169,7 @@ zs_get_chan_addr(zsc_unit, channel)
/* Find PROM mappings (for console support). */
-static int zsinited = 0; /* 0 = not, 1 = inited, not attached, 2= attached */
+int zsinited = 0; /* 0 = not, 1 = inited, not attached, 2= attached */
void
zs_init()
@@ -187,7 +219,7 @@ static u_char zs_init_reg[16] = {
/* Definition of the driver for autoconfig. */
static int zsc_match(struct device *, void *, void *);
static void zsc_attach(struct device *, struct device *, void *);
-static int zsc_print(void *aux, const char *name);
+static int zsc_print(void *, const char *name);
struct cfattach zsc_ca = {
sizeof(struct zsc_softc), zsc_match, zsc_attach
@@ -197,8 +229,8 @@ struct cfdriver zsc_cd = {
NULL, "zsc", DV_DULL
};
-int zshard(void *);
-int zssoft(void *);
+int zshard(void *);
+int zssoft(void *);
/*
@@ -213,22 +245,6 @@ zsc_match(parent, vcf, aux)
return 1;
}
-static int
-zsc_print(aux, name)
- void *aux;
- const char *name;
-{
- struct zsc_attach_args *args = aux;
-
- if (name != NULL)
- printf("%s: ", name);
-
- if (args->channel != -1)
- printf(" channel %d", args->channel);
-
- return UNCONF;
-}
-
/*
* Attach a found zs.
*
@@ -244,12 +260,13 @@ zsc_attach(parent, self, aux)
struct zsc_softc *zsc = (void *) self;
struct zsc_attach_args zsc_args;
volatile struct zschan *zc;
+ struct xzs_chanstate *xcs;
struct zs_chanstate *cs;
int zsc_unit, channel;
- int reset, s;
- int chip = 0; /* XXX quiet bogus gcc warning */
+ int s, chip, theflags;
- if (!zsinited) zs_init();
+ if (!zsinited)
+ zs_init();
zsinited = 2;
zsc_unit = zsc->zsc_dev.dv_unit;
@@ -258,52 +275,138 @@ zsc_attach(parent, self, aux)
if (zsaddr[zsc_unit] == NULL)
panic("zs_attach: zs%d not mapped", zsc_unit);
+ chip = 0;
+#ifdef DEBUG
+ printf(" chip type %d",chip);
+#endif
+ printf("\n");
+
/*
* Initialize software state for each channel.
*/
for (channel = 0; channel < 2; channel++) {
- cs = &zsc->zsc_cs[channel];
-
- zc = zs_get_chan_addr(zsc_unit, channel);
- cs->cs_reg_csr = &zc->zc_csr;
- cs->cs_reg_data = &zc->zc_data;
+ zsc_args.channel = channel;
+ zsc_args.hwflags = zs_hwflags[zsc_unit][channel];
+ xcs = &zsc->xzsc_xcs_store[channel];
+ cs = &xcs->xzs_cs;
+ zsc->zsc_cs[channel] = cs;
cs->cs_channel = channel;
cs->cs_private = NULL;
cs->cs_ops = &zsops_null;
- /* Define BAUD rate clock for the MI code. */
- cs->cs_pclk_div16 = mac68k_machine.sccClkConst*2;
- cs->cs_csource = 0;
- cs->cs_psource = 0;
+ zc = zs_get_chan_addr(zsc_unit, channel);
+ cs->cs_reg_csr = &zc->zc_csr;
+ cs->cs_reg_data = &zc->zc_data;
- cs->cs_defspeed = zs_defspeed[zsc_unit][channel];
+ if (channel == 0) /* Double check interupts are off */
+ zs_write_reg(cs, 9, 0);
bcopy(zs_init_reg, cs->cs_creg, 16);
bcopy(zs_init_reg, cs->cs_preg, 16);
- /*
- * Clear the master interrupt enable.
- * The INTENA is common to both channels,
- * so just do it on the A channel.
- */
+ /* Current BAUD rate generator clock. */
+ cs->cs_brg_clk = ZS_STD_BRG; /* RTxC is 230400*16, so use 230400 */
+ cs->cs_defspeed = zs_defspeed[zsc_unit][channel];
+ cs->cs_defcflag = zs_def_cflag;
+
+ /* Make these correspond to cs_defcflag (-crtscts) */
+ cs->cs_rr0_dcd = ZSRR0_DCD;
+ cs->cs_rr0_cts = 0;
+ cs->cs_wr5_dtr = ZSWR5_DTR;
+ cs->cs_wr5_rts = 0;
+
+#ifdef __notyet__
+ cs->cs_slave_type = ZS_SLAVE_NONE;
+#endif
+
+ /* Define BAUD rate stuff. */
+ xcs->cs_clocks[0].clk = ZS_STD_BRG * 16;
+ xcs->cs_clocks[0].flags = ZSC_RTXBRG;
+ xcs->cs_clocks[1].flags =
+ ZSC_RTXBRG | ZSC_RTXDIV | ZSC_VARIABLE | ZSC_EXTERN;
+ xcs->cs_clocks[2].flags = ZSC_TRXDIV | ZSC_VARIABLE;
+ xcs->cs_clock_count = 3;
+
if (channel == 0) {
- zs_write_reg(cs, 9, 0);
+ theflags = mac68k_machine.modem_flags;
+ xcs->cs_clocks[1].clk = mac68k_machine.modem_dcd_clk;
+ xcs->cs_clocks[2].clk = mac68k_machine.modem_cts_clk;
+ } else {
+ theflags = mac68k_machine.print_flags;
+ xcs->cs_clocks[1].flags = ZSC_VARIABLE;
+ /*
+ * Yes, we aren't defining ANY clock source enables for
+ * the printer's DCD clock in. The hardware won't let
+ * us use it. But a clock will freak out the chip, so
+ * we let you set it, telling us to bar interrupts on
+ * the line.
+ */
+ xcs->cs_clocks[1].clk = mac68k_machine.print_dcd_clk;
+ xcs->cs_clocks[2].clk = mac68k_machine.print_cts_clk;
+ }
+ if (xcs->cs_clocks[1].clk)
+ zsc_args.hwflags |= ZS_HWFLAG_NO_DCD;
+ if (xcs->cs_clocks[2].clk)
+ zsc_args.hwflags |= ZS_HWFLAG_NO_CTS;
+
+#ifdef DEBUG
+ printf("zsc%d channel %d: d_speed %6d DCD clk %ld CTS clk %ld",
+ zsc_unit, channel, cs->cs_defspeed,
+ xcs->cs_clocks[1].clk, xcs->cs_clocks[2].clk);
+#endif
+
+ /* Set defaults in our "extended" chanstate. */
+ xcs->cs_csource = 0;
+ xcs->cs_psource = 0;
+ xcs->cs_cclk_flag = 0; /* Nothing fancy by default */
+ xcs->cs_pclk_flag = 0;
+
+ if (theflags & ZSMAC_RAW) {
+ zsc_args.hwflags |= ZS_HWFLAG_RAW;
+#ifdef DEBUG
+ printf(" (raw defaults)");
+#endif
+ }
- chip = 0; /* We'll turn chip checking on post 1.2 */
- printf(" chip type %d \n",chip);
+ /*
+ * XXX - This might be better done with a "stub" driver
+ * (to replace zstty) that ignores LocalTalk for now.
+ */
+ if (theflags & ZSMAC_LOCALTALK) {
+ printf(" shielding from LocalTalk");
+ cs->cs_defspeed = 1;
+ cs->cs_creg[ZSRR_BAUDLO] = cs->cs_preg[ZSRR_BAUDLO] = 0xff;
+ cs->cs_creg[ZSRR_BAUDHI] = cs->cs_preg[ZSRR_BAUDHI] = 0xff;
+ zs_write_reg(cs, ZSRR_BAUDLO, 0xff);
+ zs_write_reg(cs, ZSRR_BAUDHI, 0xff);
+ /*
+ * If we might have LocalTalk, then make sure we have
+ * the Baud rate low-enough to not do any damage.
+ */
}
- cs->cs_chip = chip;
+
+ /*
+ * We used to disable chip interrupts here, but we now
+ * do that in zscnprobe, just in case MacOS left the chip on.
+ */
+
+ xcs->cs_chip = chip;
+
+ /* Stash away a copy of the final H/W flags. */
+ xcs->cs_hwflags = zsc_args.hwflags;
+
+#ifdef DEBUG
+ printf("\n");
+#endif
/*
* Look for a child driver for this channel.
* The child attach will setup the hardware.
*/
- zsc_args.channel = channel;
- zsc_args.hwflags = zs_hwflags[zsc_unit][channel];
- if (!config_found(self, (void *) &zsc_args, zsc_print)) {
+ if (!config_found(self, (void *)&zsc_args, zsc_print)) {
/* No sub-driver. Just reset it. */
- reset = (channel == 0) ?
+ u_char reset = (channel == 0) ?
ZSWR9_A_RESET : ZSWR9_B_RESET;
s = splzs();
zs_write_reg(cs, 9, reset);
@@ -311,161 +414,111 @@ zsc_attach(parent, self, aux)
}
}
+ if (current_mac_model->class == MACH_CLASSAV) {
+ add_psc_lev4_intr(2, zshard, zsc);
+ add_psc_lev4_intr(3, zshard, zsc);
+ } else {
+ intr_establish(zshard, zsc, ZSHARD_PRI);
+ }
+
+ /* XXX - Now safe to install interrupt handlers. */
+
/*
* Set the master interrupt enable and interrupt vector.
* (common to both channels, do it on A)
*/
- cs = &zsc->zsc_cs[0];
+ cs = zsc->zsc_cs[0];
s = splzs();
/* interrupt vector */
zs_write_reg(cs, 2, zs_init_reg[2]);
/* master interrupt control (enable) */
zs_write_reg(cs, 9, zs_init_reg[9]);
+ nzsc_attached++;
splx(s);
}
-void
-zstty_mdattach(zsc, zst, cs, tp)
- struct zsc_softc *zsc;
- struct zstty_softc *zst;
- struct zs_chanstate *cs;
- struct tty *tp;
+static int
+zsc_print(aux, name)
+ void *aux;
+ const char *name;
{
- int theflags;
-
- zst->zst_resetdef = 0;
- cs->cs_clock_count = 3; /* internal + externals */
- cs->cs_cclk_flag = 0; /* Not doing anything fancy by default */
- cs->cs_pclk_flag = 0;
- cs->cs_clocks[0].clk = mac68k_machine.sccClkConst*32;
- cs->cs_clocks[0].flags = ZSC_RTXBRG; /* allowing divide by 16 will
- melt the driver! */
-
- cs->cs_clocks[1].flags = ZSC_RTXBRG | ZSC_RTXDIV | ZSC_VARIABLE | ZSC_EXTERN;
- cs->cs_clocks[2].flags = ZSC_TRXDIV | ZSC_VARIABLE;
- if (zst->zst_dev.dv_unit == 0) {
- theflags = mac68k_machine.modem_flags;
- cs->cs_clocks[1].clk = mac68k_machine.modem_dcd_clk;
- cs->cs_clocks[2].clk = mac68k_machine.modem_cts_clk;
- } else if (zst->zst_dev.dv_unit == 1) {
- theflags = mac68k_machine.print_flags;
- cs->cs_clocks[1].flags = ZSC_VARIABLE;
- /*
- * Yes, we aren't defining ANY clock source enables for the
- * printer's DCD clock in. The hardware won't let us
- * use it. But a clock will freak out the chip, so we
- * let you set it, telling us to bar interrupts on the line.
- */
- cs->cs_clocks[1].clk = mac68k_machine.print_dcd_clk;
- cs->cs_clocks[2].clk = mac68k_machine.print_cts_clk;
- } else
- theflags = 0;
-
- if (cs->cs_clocks[1].clk)
- zst->zst_hwflags |= ZS_HWFLAG_IGDCD;
- if (cs->cs_clocks[2].clk)
- zst->zst_hwflags |= ZS_HWFLAG_IGCTS;
-
- if (theflags & ZSMAC_RAW) {
- zst->zst_cflag = ZSTTY_RAW_CFLAG;
- zst->zst_iflag = ZSTTY_RAW_IFLAG;
- zst->zst_lflag = ZSTTY_RAW_LFLAG;
- zst->zst_oflag = ZSTTY_RAW_OFLAG;
- printf(" (raw defaults)");
- }
- if (theflags & ZSMAC_LOCALTALK) {
- printf(" shielding from LocalTalk");
- zst->zst_ospeed = tp->t_ospeed = 1;
- zst->zst_ispeed = tp->t_ispeed = 1;
- cs->cs_defspeed = 1;
- cs->cs_creg[ZSRR_BAUDLO] = cs->cs_preg[ZSRR_BAUDLO] = 0xff;
- cs->cs_creg[ZSRR_BAUDHI] = cs->cs_preg[ZSRR_BAUDHI] = 0xff;
- zs_write_reg(cs, ZSRR_BAUDLO, 0xff);
- zs_write_reg(cs, ZSRR_BAUDHI, 0xff);
- /*
- * If we might have LocalTalk, then make sure we have the
- * Baud rate low-enough to not do any damage.
- */
- }
+ struct zsc_attach_args *args = aux;
- /* For the mac, we have rtscts = check CTS for output control, no
- * input control. mdmbuf means check DCD for output, and use DTR
- * for input control. mdmbuf & rtscts means use CTS for output
- * control, and DTR for input control. */
+ if (name != NULL)
+ printf("%s: ", name);
- zst->zst_hwimasks[1] = 0;
- zst->zst_hwimasks[2] = ZSWR5_DTR;
- zst->zst_hwimasks[3] = ZSWR5_DTR;
+ if (args->channel != -1)
+ printf(" channel %d", args->channel);
+ return UNCONF;
}
int
-zsmdioctl(tp, com, data, flag, p)
- struct tty *tp;
- u_long com;
+zsmdioctl(cs, cmd, data)
+ struct zs_chanstate *cs;
+ u_long cmd;
caddr_t data;
- int flag;
- struct proc *p;
{
- return (-1);
+ switch (cmd) {
+ default:
+ return (-1);
+ }
+ return (0);
}
void
zsmd_setclock(cs)
struct zs_chanstate *cs;
{
+ struct xzs_chanstate *xcs = (void *)cs;
+
if (cs->cs_channel != 0)
return;
+
/*
* If the new clock has the external bit set, then select the
* external source.
*/
- via_set_modem((cs->cs_pclk_flag & ZSC_EXTERN) ? 1 : 0);
+ via_set_modem((xcs->cs_pclk_flag & ZSC_EXTERN) ? 1 : 0);
}
+static int zssoftpending;
+
+/*
+ * Do the minimum work to pull data off of the chip and queue it up
+ * for later processing.
+ */
int
zshard(arg)
void *arg;
{
- struct zsc_softc *zsc;
- int unit, rval;
-#ifdef ZSMACDEBUG
- itecnputc(mac68k_zsdev, 'Z');
-#endif
-
- rval = 0;
- unit = zsc_cd.cd_ndevs;
- while (--unit >= 0) {
- zsc = zsc_cd.cd_devs[unit];
- if (zsc != NULL) {
- rval |= zsc_intr_hard(zsc);
- }
- }
-#ifdef ZSMACDEBUG
- itecnputc(mac68k_zsdev, '\n');
-#endif
- return (rval);
-}
+ struct zsc_softc *zsc = (struct zsc_softc *)arg;
+ int rval;
-int zssoftpending;
+ if (zsc == NULL)
+ return 0;
-void
-zsc_req_softint(zsc)
- struct zsc_softc *zsc;
-{
- if (zssoftpending == 0) {
+ rval = 0;
+ rval |= zsc_intr_hard(zsc);
+ if ((zsc->zsc_cs[0]->cs_softreq) || (zsc->zsc_cs[1]->cs_softreq)) {
+ /* zsc_req_softint(zsc); */
/* We are at splzs here, so no need to lock. */
- zssoftpending = ZSSOFT_PRI;
- /* isr_soft_request(ZSSOFT_PRI); */
- setsoftserial();
+ if (zssoftpending == 0) {
+ zssoftpending = 1;
+ setsoftserial();
+ }
}
+ return (rval);
}
+/*
+ * Similar scheme as for zshard (look at all of them)
+ */
int
zssoft(arg)
void *arg;
{
- struct zsc_softc *zsc;
- int unit;
+ register struct zsc_softc *zsc;
+ register int unit;
/* This is not the only ISR on this IPL. */
if (zssoftpending == 0)
@@ -473,32 +526,290 @@ zssoft(arg)
/*
* The soft intr. bit will be set by zshard only if
- * the variable zssoftpending is zero. The order of
- * these next two statements prevents our clearing
- * the soft intr bit just after zshard has set it.
+ * the variable zssoftpending is zero.
*/
-/* isr_soft_clear(ZSSOFT_PRI); */
zssoftpending = 0;
- /* Do ttya/ttyb first, because they go faster. */
- unit = zsc_cd.cd_ndevs;
- while (--unit >= 0) {
+ for (unit = 0; unit < zsc_cd.cd_ndevs; ++unit) {
zsc = zsc_cd.cd_devs[unit];
- if (zsc != NULL) {
- (void) zsc_intr_soft(zsc);
- }
+ if (zsc == NULL)
+ continue;
+ (void) zsc_intr_soft(zsc);
}
return (1);
}
+#ifndef ZS_TOLERANCE
+#define ZS_TOLERANCE 51
+/* 5% in tenths of a %, plus 1 so that exactly 5% will be ok. */
+#endif
+
/*
- * Read or write the chip with suitable delays.
+ * check out a rate for acceptability from the internal clock
+ * source. Used in console config to validate a requested
+ * default speed. Placed here so that all the speed checking code is
+ * in one place.
+ *
+ * != 0 means ok.
*/
-#define ZS_DELAY()
+int
+zs_cn_check_speed(bps)
+ int bps; /* target rate */
+{
+ int tc, rate;
+
+ tc = BPS_TO_TCONST(ZS_STD_BRG, bps);
+ if (tc < 0)
+ return 0;
+ rate = TCONST_TO_BPS(ZS_STD_BRG, tc);
+ if (ZS_TOLERANCE > abs(((rate - bps)*1000)/bps))
+ return 1;
+ else
+ return 0;
+}
+
/*
- * MacII hardware has the delay built in. No need for extra delay. :-)
+ * Search through the signal sources in the channel, and
+ * pick the best one for the baud rate requested. Return
+ * a -1 if not achievable in tolerance. Otherwise return 0
+ * and fill in the values.
+ *
+ * This routine draws inspiration from the Atari port's zs.c
+ * driver in NetBSD 1.1 which did the same type of source switching.
+ * Tolerance code inspired by comspeed routine in isa/com.c.
+ *
+ * By Bill Studenmund, 1996-05-12
*/
+int
+zs_set_speed(cs, bps)
+ struct zs_chanstate *cs;
+ int bps; /* bits per second */
+{
+ struct xzs_chanstate *xcs = (void *) cs;
+ int i, tc, tc0 = 0, tc1, s, sf = 0;
+ int src, rate0, rate1, err, tol;
+
+ if (bps == 0)
+ return (0);
+
+ src = -1; /* no valid source yet */
+ tol = ZS_TOLERANCE;
+
+ /*
+ * Step through all the sources and see which one matches
+ * the best. A source has to match BETTER than tol to be chosen.
+ * Thus if two sources give the same error, the first one will be
+ * chosen. Also, allow for the possability that one source might run
+ * both the BRG and the direct divider (i.e. RTxC).
+ */
+ for (i=0; i < xcs->cs_clock_count; i++) {
+ if (xcs->cs_clocks[i].clk <= 0)
+ continue; /* skip non-existant or bad clocks */
+ if (xcs->cs_clocks[i].flags & ZSC_BRG) {
+ /* check out BRG at /16 */
+ tc1 = BPS_TO_TCONST(xcs->cs_clocks[i].clk >> 4, bps);
+ if (tc1 >= 0) {
+ rate1 = TCONST_TO_BPS(xcs->cs_clocks[i].clk >> 4, tc1);
+ err = abs(((rate1 - bps)*1000)/bps);
+ if (err < tol) {
+ tol = err;
+ src = i;
+ sf = xcs->cs_clocks[i].flags & ~ZSC_DIV; tc0 = tc1;
+ rate0 = rate1;
+ }
+ }
+ }
+ if (xcs->cs_clocks[i].flags & ZSC_DIV) {
+ /*
+ * Check out either /1, /16, /32, or /64
+ * Note: for /1, you'd better be using a synchronized
+ * clock!
+ */
+ int b0 = xcs->cs_clocks[i].clk, e0 = abs(b0-bps);
+ int b1 = b0 >> 4, e1 = abs(b1-bps);
+ int b2 = b1 >> 1, e2 = abs(b2-bps);
+ int b3 = b2 >> 1, e3 = abs(b3-bps);
+
+ if (e0 < e1 && e0 < e2 && e0 < e3) {
+ err = e0;
+ rate1 = b0;
+ tc1 = ZSWR4_CLK_X1;
+ } else if (e0 > e1 && e1 < e2 && e1 < e3) {
+ err = e1;
+ rate1 = b1;
+ tc1 = ZSWR4_CLK_X16;
+ } else if (e0 > e2 && e1 > e2 && e2 < e3) {
+ err = e2;
+ rate1 = b2;
+ tc1 = ZSWR4_CLK_X32;
+ } else {
+ err = e3;
+ rate1 = b3;
+ tc1 = ZSWR4_CLK_X64;
+ }
+
+ err = (err * 1000)/bps;
+ if (err < tol) {
+ tol = err;
+ src = i;
+ sf = xcs->cs_clocks[i].flags & ~ZSC_BRG;
+ tc0 = tc1;
+ rate0 = rate1;
+ }
+ }
+ }
+#ifdef ZSMACDEBUG
+ zsprintf("Checking for rate %d. Found source #%d.\n",bps, src);
+#endif
+ if (src == -1)
+ return (EINVAL); /* no can do */
+
+ /*
+ * The M.I. layer likes to keep cs_brg_clk current, even though
+ * we are the only ones who should be touching the BRG's rate.
+ *
+ * Note: we are assuming that any ZSC_EXTERN signal source comes in
+ * on the RTxC pin. Correct for the mac68k obio zsc.
+ */
+ if (sf & ZSC_EXTERN)
+ cs->cs_brg_clk = xcs->cs_clocks[i].clk >> 4;
+ else
+ cs->cs_brg_clk = ZS_STD_BRG;
+
+ /*
+ * Now we have a source, so set it up.
+ */
+ s = splzs();
+ xcs->cs_psource = src;
+ xcs->cs_pclk_flag = sf;
+ bps = rate0;
+ if (sf & ZSC_BRG) {
+ cs->cs_preg[4] = ZSWR4_CLK_X16;
+ cs->cs_preg[11]= ZSWR11_RXCLK_BAUD | ZSWR11_TXCLK_BAUD;
+ if (sf & ZSC_PCLK) {
+ cs->cs_preg[14] = ZSWR14_BAUD_ENA | ZSWR14_BAUD_FROM_PCLK;
+ } else {
+ cs->cs_preg[14] = ZSWR14_BAUD_ENA;
+ }
+ tc = tc0;
+ } else {
+ cs->cs_preg[4] = tc0;
+ if (sf & ZSC_RTXDIV) {
+ cs->cs_preg[11] = ZSWR11_RXCLK_RTXC | ZSWR11_TXCLK_RTXC; } else {
+ cs->cs_preg[11] = ZSWR11_RXCLK_TRXC | ZSWR11_TXCLK_TRXC; }
+ cs->cs_preg[14]= 0;
+ tc = 0xffff;
+ }
+ /* Set the BAUD rate divisor. */
+ cs->cs_preg[12] = tc;
+ cs->cs_preg[13] = tc >> 8;
+ splx(s);
+
+#ifdef ZSMACDEBUG
+ zsprintf("Rate is %7d, tc is %7d, source no. %2d, flags %4x\n", \
+ bps, tc, src, sf);
+ zsprintf("Registers are: 4 %x, 11 %x, 14 %x\n\n",
+ cs->cs_preg[4], cs->cs_preg[11], cs->cs_preg[14]);
+#endif
+
+ cs->cs_preg[5] |= ZSWR5_RTS; /* Make sure the drivers are on! */
+
+ /* Caller will stuff the pending registers. */
+ return (0);
+}
+
+int
+zs_set_modes(cs, cflag)
+ struct zs_chanstate *cs;
+ int cflag; /* bits per second */
+{
+ struct xzs_chanstate *xcs = (void*)cs;
+ int s;
+
+ /*
+ * Make sure we don't enable hfc on a signal line we're ignoring.
+ * As we enable CTS interrupts only if we have CRTSCTS or CDTRCTS,
+ * this code also effectivly turns off ZSWR15_CTS_IE.
+ *
+ * Also, disable DCD interrupts if we've been told to ignore
+ * the DCD pin. Happens on mac68k because the input line for
+ * DCD can also be used as a clock input. (Just set CLOCAL.)
+ *
+ * If someone tries to turn an invalid flow mode on, Just Say No
+ * (Suggested by gwr)
+ */
+#if 0
+ if ((cflag & CDTRCTS) && (cflag & (CRTSCTS | MDMBUF)))
+ return (EINVAL);
+#endif
+ if (xcs->cs_hwflags & ZS_HWFLAG_NO_DCD) {
+ if (cflag & MDMBUF)
+ return (EINVAL);
+ cflag |= CLOCAL;
+ }
+#if 0
+ if ((xcs->cs_hwflags & ZS_HWFLAG_NO_CTS) && (cflag & (CRTSCTS | CDTRCTS)))
+ return (EINVAL);
+#endif
+
+ /*
+ * Output hardware flow control on the chip is horrendous:
+ * if carrier detect drops, the receiver is disabled, and if
+ * CTS drops, the transmitter is stoped IN MID CHARACTER!
+ * Therefore, NEVER set the HFC bit, and instead use the
+ * status interrupt to detect CTS changes.
+ */
+ s = splzs();
+ if ((cflag & (CLOCAL | MDMBUF)) != 0)
+ cs->cs_rr0_dcd = 0;
+ else
+ cs->cs_rr0_dcd = ZSRR0_DCD;
+ /*
+ * The mac hardware only has one output, DTR (HSKo in Mac
+ * parlance). In HFC mode, we use it for the functions
+ * typically served by RTS and DTR on other ports, so we
+ * have to fake the upper layer out some.
+ *
+ * CRTSCTS we use CTS as an input which tells us when to shut up.
+ * We make no effort to shut up the other side of the connection.
+ * DTR is used to hang up the modem.
+ *
+ * In CDTRCTS, we use CTS to tell us to stop, but we use DTR to
+ * shut up the other side.
+ */
+ if ((cflag & CRTSCTS) != 0){
+ cs->cs_wr5_dtr = ZSWR5_DTR;
+ cs->cs_wr5_rts = 0;
+ cs->cs_rr0_cts = ZSRR0_CTS;
+#if 0
+ } else if ((cflag & CDTRCTS) != 0) {
+ cs->cs_wr5_dtr = 0;
+ cs->cs_wr5_rts = ZSWR5_DTR;
+ cs->cs_rr0_cts = ZSRR0_CTS;
+#endif
+ } else if ((cflag & MDMBUF) != 0) {
+ cs->cs_wr5_dtr = 0;
+ cs->cs_wr5_rts = ZSWR5_DTR;
+ cs->cs_rr0_cts = ZSRR0_DCD;
+ } else {
+ cs->cs_wr5_dtr = ZSWR5_DTR;
+ cs->cs_wr5_rts = 0;
+ cs->cs_rr0_cts = 0;
+ }
+ splx(s);
+
+ /* Caller will stuff the pending registers. */
+ return(0);
+}
+
+/*
+ * Read or write the chip with suitable delays.
+ * MacII hardware has the delay built in.
+ * No need for extra delay. :-) However, some clock-chiped
+ * macs, or zsc's on serial add-on boards might need it.
+ */
+#define ZS_DELAY()
u_char
zs_read_reg(cs, reg)
@@ -528,30 +839,32 @@ zs_write_reg(cs, reg, val)
u_char zs_read_csr(cs)
struct zs_chanstate *cs;
{
- register u_char v;
+ u_char val;
- v = (*cs->cs_reg_csr) ^ ZSRR0_CTS;
- /* make up for the fact CTS is wired backwards */
+ val = *cs->cs_reg_csr;
ZS_DELAY();
- return v;
+ /* make up for the fact CTS is wired backwards */
+ val ^= ZSRR0_CTS;
+ return val;
}
-u_char zs_read_data(cs)
+void zs_write_csr(cs, val)
struct zs_chanstate *cs;
+ register u_char val;
{
- register u_char v;
-
- v = *cs->cs_reg_data;
+ /* Note, the csr does not write CTS... */
+ *cs->cs_reg_csr = val;
ZS_DELAY();
- return v;
}
-void zs_write_csr(cs, val)
+u_char zs_read_data(cs)
struct zs_chanstate *cs;
- u_char val;
{
- *cs->cs_reg_csr = val;
+ register u_char val;
+
+ val = *cs->cs_reg_data;
ZS_DELAY();
+ return val;
}
void zs_write_data(cs, val)
@@ -563,8 +876,11 @@ void zs_write_data(cs, val)
}
/****************************************************************
- * Console support functions (Originally Sun3 specific!)
- * Now works w/ just mac68k port!
+ * Console support functions (mac68k specific!)
+ * Note: this code is allowed to know about the layout of
+ * the chip registers, and uses that to keep things simple.
+ * XXX - I think I like the mvme167 code better. -gwr
+ * XXX - Well :-P :-) -wrs
****************************************************************/
#define zscnpollc nullcnpollc
@@ -587,27 +903,38 @@ extern int zsopen( dev_t dev, int flags, int mode, struct proc *p);
static void
zscnsetup()
{
- struct zs_chanstate cs;
+ struct xzs_chanstate xcs;
+ struct zs_chanstate *cs;
struct zschan *zc;
int tconst, s;
/* Setup temporary chanstate. */
- bzero((caddr_t)&cs, sizeof(cs));
+ bzero((caddr_t)&xcs, sizeof(xcs));
+ cs = &xcs.xzs_cs;
zc = zs_conschan;
- cs.cs_reg_csr = &zc->zc_csr;
- cs.cs_reg_data = &zc->zc_data;
- cs.cs_channel = zs_consunit;
-
- bcopy(zs_init_reg, cs.cs_preg, 16);
- tconst = BPS_TO_TCONST(mac68k_machine.sccClkConst*2, zs_defspeed[0][zs_consunit]);
- cs.cs_preg[5] |= ZSWR5_DTR | ZSWR5_RTS;
- cs.cs_preg[1] = 0; /* don't enable interrupts */
- cs.cs_preg[12] = tconst;
- cs.cs_preg[13] = tconst >> 8;
-
- s = splhigh();
- zs_loadchannelregs(&cs);
- splx(s);
+ cs->cs_reg_csr = &zc->zc_csr;
+ cs->cs_reg_data = &zc->zc_data;
+ cs->cs_channel = zs_consunit;
+ cs->cs_brg_clk = ZS_STD_BRG;
+
+ bcopy(zs_init_reg, cs->cs_preg, 16);
+ cs->cs_preg[5] |= ZSWR5_DTR | ZSWR5_RTS;
+ cs->cs_preg[15] = ZSWR15_BREAK_IE;
+ tconst = BPS_TO_TCONST(cs->cs_brg_clk,
+ zs_defspeed[0][zs_consunit]);
+ cs->cs_preg[12] = tconst;
+ cs->cs_preg[13] = tconst >> 8;
+ /* can't use zs_set_speed as we haven't set up the
+ * signal sources, and it's not worth it for now
+ */
+
+ /*
+ * As zs_loadchannelregs doesn't touch reg 9 (interupt control),
+ * we won't accidentally turn on interupts below
+ */
+ s = splhigh();
+ zs_loadchannelregs(cs);
+ splx(s);
}
/*
@@ -620,81 +947,85 @@ zscnsetup()
* (the video, better than CN_NORMAL), and CN_REMOTE (pick me!)
*
* As the mac's a bit different, we do extra work here. We mainly check
- * to see if we have serial echo going on, and if the tty's are supposed
- * to default to raw or not.
+ * to see if we have serial echo going on. Also chould check for default
+ * speeds.
*/
void
zscnprobe(struct consdev * cp)
{
- extern u_long IOBase;
- int maj, unit;
-
- for (maj = 0; maj < nchrdev; maj++) {
- if (cdevsw[maj].d_open == zsopen) {
- break;
- }
- }
- if (maj == nchrdev) {
- /* no console entry for us */
- if (mac68k_machine.serial_boot_echo) {
- mac68k_set_io_offsets(IOBase);
- zs_conschan = (struct zschan *) -1; /* dummy flag for zs_init() */
- zs_consunit = 1;
- zs_hwflags[0][zs_consunit] = ZS_HWFLAG_CONSOLE;
- zs_init();
- zscnsetup();
- }
- return;
- }
-
- cp->cn_pri = CN_NORMAL; /* Lower than CN_INTERNAL */
- if (mac68k_machine.serial_console != 0) {
- cp->cn_pri = CN_REMOTE; /* Higher than CN_INTERNAL */
- mac68k_machine.serial_boot_echo =0;
- }
-
- unit = (mac68k_machine.serial_console == 1) ? 0 : 1;
- zs_consunit = unit;
-
- mac68k_zsdev = cp->cn_dev = makedev(maj, unit);
-
- if (mac68k_machine.serial_boot_echo) {
- /*
- * at this point, we know that we don't have a serial
- * console, but are doing echo
- */
- mac68k_set_io_offsets(IOBase);
- zs_conschan = (struct zschan *) -1; /* dummy flag for zs_init() */
+ extern u_long IOBase;
+ int maj, unit, i;
+
+ for (maj = 0; maj < nchrdev; maj++) {
+ if (cdevsw[maj].d_open == zsopen) {
+ break;
+ }
+ }
+ if (maj != nchrdev) {
+ cp->cn_pri = CN_NORMAL; /* Lower than CN_INTERNAL */
+ if (mac68k_machine.serial_console != 0) {
+ cp->cn_pri = CN_REMOTE; /* Higher than CN_INTERNAL */
+ mac68k_machine.serial_boot_echo =0;
+ }
+
+ unit = (mac68k_machine.serial_console == 1) ? 0 : 1;
+ zs_consunit = unit;
+ zs_conschan = (struct zschan *) -1; /* dummy flag for zs_init() */
+
+ mac68k_zsdev = cp->cn_dev = makedev(maj, unit);
+ }
+ if (mac68k_machine.serial_boot_echo) {
+ /*
+ * at this point, we know that we don't have a serial
+ * console, but are doing echo
+ */
+ zs_conschan = (struct zschan *) -1; /* dummy flag for zs_init() */
zs_consunit = 1;
zs_hwflags[0][zs_consunit] = ZS_HWFLAG_CONSOLE;
- zs_init();
- zscnsetup();
- }
- return;
+ }
+
+ if ((i = mac68k_machine.modem_d_speed) > 0) {
+ if (zs_cn_check_speed(i))
+ zs_defspeed[0][0] = i;
+ }
+ if ((i = mac68k_machine.print_d_speed) > 0) {
+ if (zs_cn_check_speed(i))
+ zs_defspeed[0][1] = i;
+ }
+ mac68k_set_io_offsets(IOBase);
+ zs_init();
+ /*
+ * zsinit will set up the addresses of the scc. It will also, if
+ * zs_conschan != 0, calculate the new address of the conschan for
+ * unit zs_consunit. So if we are (or think we are) going to use the
+ * chip for console I/O, we just set up the internal addresses for it.
+ *
+ * Now turn off interrupts for the chip. Note: this code piece is the
+ * only vestage of the NetBSD 1.0 ser driver. :-)
+ */
+ unit = sccA[2]; /* reset reg. access */
+ unit = sccA[0];
+ sccA[2] = 9; sccA[2] = 0; /* write 0 to reg. 9, clearing MIE */
+ sccA[2] = ZSWR0_CLR_INTR; unit = sccA[2]; /* reset any pending ints. */
+ sccA[0] = ZSWR0_CLR_INTR; unit = sccA[0];
+
+ if (mac68k_machine.serial_boot_echo)
+ zscnsetup();
+ return;
}
void
zscninit(struct consdev * cp)
{
- extern u_long IOBase;
- int chan = minor(cp->cn_dev & 1);
-
- mac68k_set_io_offsets(IOBase);
- zs_conschan = (struct zschan *) -1;
- zs_consunit = chan;
zs_hwflags[0][zs_consunit] = ZS_HWFLAG_CONSOLE;
-#ifdef ZS_CONSOLE_ABORT
- zs_hwflags[0][zs_consunit] |= ZS_HWFLAG_CONABRT;
-#endif
- zs_init();
- /*
+ /*
* zsinit will set up the addresses of the scc. It will also, if
* zs_conschan != 0, calculate the new address of the conschan for
* unit zs_consunit. So zs_init implicitly sets zs_conschan to the right
* number. :-)
- */
- zscnsetup();
- printf("\nOpenBSD/mac68k console\n");
+ */
+ zscnsetup();
+ printf("\nOpenBSD/mac68k console\n");
}
@@ -784,27 +1115,21 @@ zscnputc(dev, c)
* Handle user request to enter kernel debugger.
*/
void
-zs_abort(zst)
- register struct zstty_softc *zst;
+zs_abort(cs)
+ struct zs_chanstate *cs;
{
- register volatile struct zschan *zc = zs_conschan;
+ volatile struct zschan *zc = zs_conschan;
int rr0;
register long wait = 0;
- if ((zst->zst_hwflags & ZS_HWFLAG_CONABRT) == 0)
- return;
-
/* Wait for end of break to avoid PROM abort. */
- /* XXX - Limit the wait? */
do {
rr0 = zc->zc_csr;
ZS_DELAY();
} while ((rr0 & ZSRR0_BREAK) && (wait++ < ZSABORT_DELAY));
- if (wait > ZSABORT_DELAY) {
- if (zst != NULL) zst->zst_hwflags &= ~ZS_HWFLAG_CONABRT;
- /* If we time out, turn off the abort ability! */
- }
+ if (wait > ZSABORT_DELAY)
+ return; /* XXX */
#ifdef DDB
if (db_console)
Debugger();