1 /* $NetBSD: zs.c,v 1.69 2009/10/26 19:55:11 martin Exp $ */ 2 3 /*- 4 * Copyright (c) 1996 The NetBSD Foundation, Inc. 5 * All rights reserved. 6 * 7 * This code is derived from software contributed to The NetBSD Foundation 8 * by Gordon W. Ross. 9 * 10 * Redistribution and use in source and binary forms, with or without 11 * modification, are permitted provided that the following conditions 12 * are met: 13 * 1. Redistributions of source code must retain the above copyright 14 * notice, this list of conditions and the following disclaimer. 15 * 2. Redistributions in binary form must reproduce the above copyright 16 * notice, this list of conditions and the following disclaimer in the 17 * documentation and/or other materials provided with the distribution. 18 * 19 * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS 20 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 21 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 22 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS 23 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 24 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 25 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 26 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 27 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 28 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 29 * POSSIBILITY OF SUCH DAMAGE. 30 */ 31 32 /* 33 * Zilog Z8530 Dual UART driver (machine-dependent part) 34 * 35 * Runs two serial lines per chip using slave drivers. 36 * Plain tty/async lines use the zs_async slave. 37 * Sun keyboard/mouse uses the zs_kbd/zs_ms slaves. 38 */ 39 40 #include <sys/cdefs.h> 41 __KERNEL_RCSID(0, "$NetBSD: zs.c,v 1.69 2009/10/26 19:55:11 martin Exp $"); 42 43 #include "opt_ddb.h" 44 #include "opt_kgdb.h" 45 46 #include <sys/param.h> 47 #include <sys/systm.h> 48 #include <sys/conf.h> 49 #include <sys/device.h> 50 #include <sys/file.h> 51 #include <sys/ioctl.h> 52 #include <sys/kernel.h> 53 #include <sys/proc.h> 54 #include <sys/tty.h> 55 #include <sys/time.h> 56 #include <sys/syslog.h> 57 #include <sys/intr.h> 58 59 #include <machine/autoconf.h> 60 #include <machine/openfirm.h> 61 #include <machine/cpu.h> 62 #include <machine/eeprom.h> 63 #include <machine/psl.h> 64 #include <machine/z8530var.h> 65 66 #include <dev/cons.h> 67 #include <dev/ic/z8530reg.h> 68 #include <dev/sun/kbd_ms_ttyvar.h> 69 #include <ddb/db_output.h> 70 71 #include <sparc64/dev/cons.h> 72 73 #include "ioconf.h" 74 #include "kbd.h" /* NKBD */ 75 #include "ms.h" /* NMS */ 76 #include "zs.h" /* NZS */ 77 78 /* Make life easier for the initialized arrays here. */ 79 #if NZS < 3 80 #undef NZS 81 #define NZS 3 82 #endif 83 84 /* 85 * Some warts needed by z8530tty.c - 86 * The default parity REALLY needs to be the same as the PROM uses, 87 * or you can not see messages done with printf during boot-up... 88 */ 89 int zs_def_cflag = (CREAD | CS8 | HUPCL); 90 91 /* 92 * The Sun provides a 4.9152 MHz clock to the ZS chips. 93 */ 94 #define PCLK (9600 * 512) /* PCLK pin input clock rate */ 95 96 #define ZS_DELAY() 97 98 /* The layout of this is hardware-dependent (padding, order). */ 99 struct zschan { 100 volatile uint8_t zc_csr; /* ctrl,status, and indirect access */ 101 uint8_t zc_xxx0; 102 volatile uint8_t zc_data; /* data */ 103 uint8_t zc_xxx1; 104 }; 105 struct zsdevice { 106 /* Yes, they are backwards. */ 107 struct zschan zs_chan_b; 108 struct zschan zs_chan_a; 109 }; 110 111 /* ZS channel used as the console device (if any) */ 112 void *zs_conschan_get, *zs_conschan_put; 113 114 /* Saved PROM mappings */ 115 static struct zsdevice *zsaddr[NZS]; 116 117 static uint8_t zs_init_reg[16] = { 118 0, /* 0: CMD (reset, etc.) */ 119 0, /* 1: No interrupts yet. */ 120 0, /* 2: IVECT */ 121 ZSWR3_RX_8 | ZSWR3_RX_ENABLE, 122 ZSWR4_CLK_X16 | ZSWR4_ONESB | ZSWR4_EVENP, 123 ZSWR5_TX_8 | ZSWR5_TX_ENABLE, 124 0, /* 6: TXSYNC/SYNCLO */ 125 0, /* 7: RXSYNC/SYNCHI */ 126 0, /* 8: alias for data port */ 127 ZSWR9_MASTER_IE | ZSWR9_NO_VECTOR, 128 0, /*10: Misc. TX/RX control bits */ 129 ZSWR11_TXCLK_BAUD | ZSWR11_RXCLK_BAUD, 130 ((PCLK/32)/9600)-2, /*12: BAUDLO (default=9600) */ 131 0, /*13: BAUDHI (default=9600) */ 132 ZSWR14_BAUD_ENA | ZSWR14_BAUD_FROM_PCLK, 133 ZSWR15_BREAK_IE, 134 }; 135 136 /* Console ops */ 137 static int zscngetc(dev_t); 138 static void zscnputc(dev_t, int); 139 static void zscnpollc(dev_t, int); 140 141 struct consdev zs_consdev = { 142 .cn_getc = zscngetc, 143 .cn_putc = zscnputc, 144 .cn_pollc = zscnpollc, 145 }; 146 147 148 /**************************************************************** 149 * Autoconfig 150 ****************************************************************/ 151 152 /* Definition of the driver for autoconfig. */ 153 static int zs_match_sbus(device_t, cfdata_t, void *); 154 static void zs_attach_sbus(device_t, device_t, void *); 155 156 static void zs_attach(struct zsc_softc *, struct zsdevice *, int); 157 static int zs_print(void *, const char *); 158 159 CFATTACH_DECL_NEW(zs, sizeof(struct zsc_softc), 160 zs_match_sbus, zs_attach_sbus, NULL, NULL); 161 162 /* Interrupt handlers. */ 163 int zscheckintr(void *); 164 static int zshard(void *); 165 static void zssoft(void *); 166 167 static int zs_get_speed(struct zs_chanstate *); 168 169 /* Console device support */ 170 static int zs_console_flags(int, int, int); 171 172 /* Power management hooks */ 173 int zs_enable(struct zs_chanstate *); 174 void zs_disable(struct zs_chanstate *); 175 176 /* from dev/ic/z8530tty.c */ 177 struct tty *zstty_get_tty_from_dev(struct device *); 178 179 /* 180 * Is the zs chip present? 181 */ 182 static int 183 zs_match_sbus(device_t parent, cfdata_t cf, void *aux) 184 { 185 struct sbus_attach_args *sa = aux; 186 187 if (strcmp(cf->cf_name, sa->sa_name) != 0) 188 return (0); 189 190 return (1); 191 } 192 193 static void 194 zs_attach_sbus(device_t parent, device_t self, void *aux) 195 { 196 struct zsc_softc *zsc = device_private(self); 197 struct sbus_attach_args *sa = aux; 198 bus_space_handle_t bh; 199 int zs_unit; 200 201 zsc->zsc_dev = self; 202 zs_unit = device_unit(self); 203 204 if (sa->sa_nintr == 0) { 205 aprint_error(": no interrupt lines\n"); 206 return; 207 } 208 209 /* Use the mapping setup by the Sun PROM if possible. */ 210 if (zsaddr[zs_unit] == NULL) { 211 /* Only map registers once. */ 212 if (sa->sa_npromvaddrs) { 213 /* 214 * We're converting from a 32-bit pointer to a 64-bit 215 * pointer. Since the 32-bit entity is negative, but 216 * the kernel is still mapped into the lower 4GB 217 * range, this needs to be zero-extended. 218 * 219 * XXXXX If we map the kernel and devices into the 220 * high 4GB range, this needs to be changed to 221 * sign-extend the address. 222 */ 223 sparc_promaddr_to_handle(sa->sa_bustag, 224 sa->sa_promvaddrs[0], &bh); 225 226 } else { 227 228 if (sbus_bus_map(sa->sa_bustag, sa->sa_slot, 229 sa->sa_offset, 230 sa->sa_size, 231 BUS_SPACE_MAP_LINEAR, 232 &bh) != 0) { 233 aprint_error(": cannot map registers\n"); 234 return; 235 } 236 } 237 zsaddr[zs_unit] = bus_space_vaddr(sa->sa_bustag, bh); 238 } 239 zsc->zsc_bustag = sa->sa_bustag; 240 zsc->zsc_dmatag = sa->sa_dmatag; 241 zsc->zsc_promunit = prom_getpropint(sa->sa_node, "slave", -2); 242 zsc->zsc_node = sa->sa_node; 243 zs_attach(zsc, zsaddr[zs_unit], sa->sa_pri); 244 } 245 246 /* 247 * Attach a found zs. 248 * 249 * USE ROM PROPERTIES port-a-ignore-cd AND port-b-ignore-cd FOR 250 * SOFT CARRIER, AND keyboard PROPERTY FOR KEYBOARD/MOUSE? 251 */ 252 static void 253 zs_attach(struct zsc_softc *zsc, struct zsdevice *zsd, int pri) 254 { 255 struct zsc_attach_args zsc_args; 256 struct zs_chanstate *cs; 257 int channel; 258 259 if (zsd == NULL) { 260 aprint_error(": configuration incomplete\n"); 261 return; 262 } 263 264 /* 265 * Initialize software state for each channel. 266 */ 267 for (channel = 0; channel < 2; channel++) { 268 struct zschan *zc; 269 struct device *child; 270 271 zsc_args.channel = channel; 272 cs = &zsc->zsc_cs_store[channel]; 273 zsc->zsc_cs[channel] = cs; 274 275 zs_lock_init(cs); 276 cs->cs_channel = channel; 277 cs->cs_private = NULL; 278 cs->cs_ops = &zsops_null; 279 cs->cs_brg_clk = PCLK / 16; 280 281 zc = (channel == 0) ? &zsd->zs_chan_a : &zsd->zs_chan_b; 282 283 zsc_args.consdev = NULL; 284 zsc_args.hwflags = zs_console_flags(zsc->zsc_promunit, 285 zsc->zsc_node, 286 channel); 287 288 if (zsc_args.hwflags & ZS_HWFLAG_CONSOLE) { 289 zsc_args.hwflags |= ZS_HWFLAG_USE_CONSDEV; 290 zsc_args.consdev = &zs_consdev; 291 } 292 293 if ((zsc_args.hwflags & ZS_HWFLAG_CONSOLE_INPUT) != 0) { 294 zs_conschan_get = zc; 295 } 296 if ((zsc_args.hwflags & ZS_HWFLAG_CONSOLE_OUTPUT) != 0) { 297 zs_conschan_put = zc; 298 } 299 300 /* Children need to set cn_dev, etc */ 301 cs->cs_reg_csr = &zc->zc_csr; 302 cs->cs_reg_data = &zc->zc_data; 303 304 memcpy(cs->cs_creg, zs_init_reg, 16); 305 memcpy(cs->cs_preg, zs_init_reg, 16); 306 307 /* XXX: Consult PROM properties for this?! */ 308 cs->cs_defspeed = zs_get_speed(cs); 309 cs->cs_defcflag = zs_def_cflag; 310 311 /* Make these correspond to cs_defcflag (-crtscts) */ 312 cs->cs_rr0_dcd = ZSRR0_DCD; 313 cs->cs_rr0_cts = 0; 314 cs->cs_wr5_dtr = ZSWR5_DTR | ZSWR5_RTS; 315 cs->cs_wr5_rts = 0; 316 317 /* 318 * Clear the master interrupt enable. 319 * The INTENA is common to both channels, 320 * so just do it on the A channel. 321 */ 322 if (channel == 0) { 323 zs_write_reg(cs, 9, 0); 324 } 325 326 /* 327 * Look for a child driver for this channel. 328 * The child attach will setup the hardware. 329 */ 330 child = config_found(zsc->zsc_dev, (void *)&zsc_args, 331 zs_print); 332 if (child == NULL) { 333 /* No sub-driver. Just reset it. */ 334 uint8_t reset = (channel == 0) ? 335 ZSWR9_A_RESET : ZSWR9_B_RESET; 336 zs_lock_chan(cs); 337 zs_write_reg(cs, 9, reset); 338 zs_unlock_chan(cs); 339 } 340 #if (NKBD > 0) || (NMS > 0) 341 /* 342 * If this was a zstty it has a keyboard 343 * property on it we need to attach the 344 * sunkbd and sunms line disciplines. 345 */ 346 if (child 347 && (device_is_a(child, "zstty")) 348 && (prom_getproplen(zsc->zsc_node, "keyboard") == 0)) { 349 struct kbd_ms_tty_attach_args kma; 350 struct tty *tp; 351 352 kma.kmta_tp = tp = zstty_get_tty_from_dev(child); 353 kma.kmta_dev = tp->t_dev; 354 kma.kmta_consdev = zsc_args.consdev; 355 356 /* Attach 'em if we got 'em. */ 357 #if (NKBD > 0) 358 if (channel == 0) { 359 kma.kmta_name = "keyboard"; 360 config_found(child, (void *)&kma, NULL); 361 } 362 #endif 363 #if (NMS > 0) 364 if (channel == 1) { 365 kma.kmta_name = "mouse"; 366 config_found(child, (void *)&kma, NULL); 367 } 368 #endif 369 } 370 #endif 371 } 372 373 /* 374 * Now safe to install interrupt handlers. Note the arguments 375 * to the interrupt handlers aren't used. Note, we only do this 376 * once since both SCCs interrupt at the same level and vector. 377 */ 378 bus_intr_establish(zsc->zsc_bustag, pri, IPL_SERIAL, zshard, zsc); 379 if (!(zsc->zsc_softintr = softint_establish(SOFTINT_SERIAL, zssoft, zsc))) 380 panic("zsattach: could not establish soft interrupt"); 381 382 evcnt_attach_dynamic(&zsc->zsc_intrcnt, EVCNT_TYPE_INTR, NULL, 383 device_xname(zsc->zsc_dev), "intr"); 384 385 386 /* 387 * Set the master interrupt enable and interrupt vector. 388 * (common to both channels, do it on A) 389 */ 390 cs = zsc->zsc_cs[0]; 391 zs_lock_chan(cs); 392 /* interrupt vector */ 393 zs_write_reg(cs, 2, zs_init_reg[2]); 394 /* master interrupt control (enable) */ 395 zs_write_reg(cs, 9, zs_init_reg[9]); 396 zs_unlock_chan(cs); 397 } 398 399 static int 400 zs_print(void *aux, const char *name) 401 { 402 struct zsc_attach_args *args = aux; 403 404 if (name != NULL) 405 aprint_normal("%s: ", name); 406 407 if (args->channel != -1) 408 aprint_normal(" channel %d", args->channel); 409 410 return (UNCONF); 411 } 412 413 static int 414 zshard(void *arg) 415 { 416 struct zsc_softc *zsc = arg; 417 int rr3, rval; 418 419 rval = 0; 420 while ((rr3 = zsc_intr_hard(zsc))) { 421 /* Count up the interrupts. */ 422 rval |= rr3; 423 zsc->zsc_intrcnt.ev_count++; 424 } 425 if (((zsc->zsc_cs[0] && zsc->zsc_cs[0]->cs_softreq) || 426 (zsc->zsc_cs[1] && zsc->zsc_cs[1]->cs_softreq)) && 427 zsc->zsc_softintr) { 428 softint_schedule(zsc->zsc_softintr); 429 } 430 return (rval); 431 } 432 433 int 434 zscheckintr(void *arg) 435 { 436 struct zsc_softc *zsc; 437 int unit, rval; 438 439 rval = 0; 440 for (unit = 0; unit < zs_cd.cd_ndevs; unit++) { 441 442 zsc = device_lookup_private(&zs_cd, unit); 443 if (zsc == NULL) 444 continue; 445 rval = (zshard((void *)zsc) || rval); 446 } 447 return (rval); 448 } 449 450 451 /* 452 * We need this only for TTY_DEBUG purposes. 453 */ 454 static void 455 zssoft(void *arg) 456 { 457 struct zsc_softc *zsc = arg; 458 459 #if 0 /* not yet */ 460 /* Make sure we call the tty layer with tty_lock held. */ 461 mutex_spin_enter(&tty_lock); 462 #endif 463 (void)zsc_intr_soft(zsc); 464 #ifdef TTY_DEBUG 465 { 466 struct zstty_softc *zst0 = zsc->zsc_cs[0]->cs_private; 467 struct zstty_softc *zst1 = zsc->zsc_cs[1]->cs_private; 468 if (zst0->zst_overflows || zst1->zst_overflows ) { 469 struct trapframe *frame = (struct trapframe *)arg; 470 471 printf("zs silo overflow from %p\n", 472 (long)frame->tf_pc); 473 } 474 } 475 #endif 476 #if 0 /* not yet */ 477 mutex_spin_exit(&tty_lock); 478 #endif 479 } 480 481 482 /* 483 * Compute the current baud rate given a ZS channel. 484 */ 485 static int 486 zs_get_speed(struct zs_chanstate *cs) 487 { 488 int tconst; 489 490 tconst = zs_read_reg(cs, 12); 491 tconst |= zs_read_reg(cs, 13) << 8; 492 return (TCONST_TO_BPS(cs->cs_brg_clk, tconst)); 493 } 494 495 /* 496 * MD functions for setting the baud rate and control modes. 497 */ 498 int 499 zs_set_speed(struct zs_chanstate *cs, int bps /* bits per second */) 500 { 501 int tconst, real_bps; 502 503 if (bps == 0) 504 return (0); 505 506 #ifdef DIAGNOSTIC 507 if (cs->cs_brg_clk == 0) 508 panic("zs_set_speed"); 509 #endif 510 511 tconst = BPS_TO_TCONST(cs->cs_brg_clk, bps); 512 if (tconst < 0) 513 return (EINVAL); 514 515 /* Convert back to make sure we can do it. */ 516 real_bps = TCONST_TO_BPS(cs->cs_brg_clk, tconst); 517 518 /* XXX - Allow some tolerance here? */ 519 if (real_bps != bps) 520 return (EINVAL); 521 522 cs->cs_preg[12] = tconst; 523 cs->cs_preg[13] = tconst >> 8; 524 525 /* Caller will stuff the pending registers. */ 526 return (0); 527 } 528 529 int 530 zs_set_modes(struct zs_chanstate *cs, int cflag) 531 { 532 533 /* 534 * Output hardware flow control on the chip is horrendous: 535 * if carrier detect drops, the receiver is disabled, and if 536 * CTS drops, the transmitter is stoped IN MID CHARACTER! 537 * Therefore, NEVER set the HFC bit, and instead use the 538 * status interrupt to detect CTS changes. 539 */ 540 zs_lock_chan(cs); 541 cs->cs_rr0_pps = 0; 542 if ((cflag & (CLOCAL | MDMBUF)) != 0) { 543 cs->cs_rr0_dcd = 0; 544 if ((cflag & MDMBUF) == 0) 545 cs->cs_rr0_pps = ZSRR0_DCD; 546 } else 547 cs->cs_rr0_dcd = ZSRR0_DCD; 548 if ((cflag & CRTSCTS) != 0) { 549 cs->cs_wr5_dtr = ZSWR5_DTR; 550 cs->cs_wr5_rts = ZSWR5_RTS; 551 cs->cs_rr0_cts = ZSRR0_CTS; 552 } else if ((cflag & CDTRCTS) != 0) { 553 cs->cs_wr5_dtr = 0; 554 cs->cs_wr5_rts = ZSWR5_DTR; 555 cs->cs_rr0_cts = ZSRR0_CTS; 556 } else if ((cflag & MDMBUF) != 0) { 557 cs->cs_wr5_dtr = 0; 558 cs->cs_wr5_rts = ZSWR5_DTR; 559 cs->cs_rr0_cts = ZSRR0_DCD; 560 } else { 561 cs->cs_wr5_dtr = ZSWR5_DTR | ZSWR5_RTS; 562 cs->cs_wr5_rts = 0; 563 cs->cs_rr0_cts = 0; 564 } 565 zs_unlock_chan(cs); 566 567 /* Caller will stuff the pending registers. */ 568 return (0); 569 } 570 571 572 /* 573 * Read or write the chip with suitable delays. 574 */ 575 576 u_char 577 zs_read_reg(struct zs_chanstate *cs, u_char reg) 578 { 579 u_char val; 580 581 *cs->cs_reg_csr = reg; 582 ZS_DELAY(); 583 val = *cs->cs_reg_csr; 584 ZS_DELAY(); 585 return (val); 586 } 587 588 void 589 zs_write_reg(struct zs_chanstate *cs, u_char reg, u_char val) 590 { 591 *cs->cs_reg_csr = reg; 592 ZS_DELAY(); 593 *cs->cs_reg_csr = val; 594 ZS_DELAY(); 595 } 596 597 u_char 598 zs_read_csr(struct zs_chanstate *cs) 599 { 600 u_char val; 601 602 val = *cs->cs_reg_csr; 603 ZS_DELAY(); 604 return (val); 605 } 606 607 void 608 zs_write_csr(struct zs_chanstate *cs, u_char val) 609 { 610 *cs->cs_reg_csr = val; 611 ZS_DELAY(); 612 } 613 614 u_char 615 zs_read_data(struct zs_chanstate *cs) 616 { 617 u_char val; 618 619 val = *cs->cs_reg_data; 620 ZS_DELAY(); 621 return (val); 622 } 623 624 void 625 zs_write_data(struct zs_chanstate *cs, u_char val) 626 { 627 *cs->cs_reg_data = val; 628 ZS_DELAY(); 629 } 630 631 /**************************************************************** 632 * Console support functions (Sun specific!) 633 * Note: this code is allowed to know about the layout of 634 * the chip registers, and uses that to keep things simple. 635 * XXX - I think I like the mvme167 code better. -gwr 636 ****************************************************************/ 637 638 extern void Debugger(void); 639 640 /* 641 * Handle user request to enter kernel debugger. 642 */ 643 void 644 zs_abort(struct zs_chanstate *cs) 645 { 646 volatile struct zschan *zc = zs_conschan_get; 647 int rr0; 648 649 /* Wait for end of break to avoid PROM abort. */ 650 /* XXX - Limit the wait? */ 651 do { 652 rr0 = zc->zc_csr; 653 ZS_DELAY(); 654 } while (rr0 & ZSRR0_BREAK); 655 656 #if defined(KGDB) 657 zskgdb(cs); 658 #elif defined(DDB) 659 { 660 extern int db_active; 661 662 if (!db_active) 663 Debugger(); 664 else 665 /* Debugger is probably hozed */ 666 callrom(); 667 } 668 #else 669 printf("stopping on keyboard abort\n"); 670 callrom(); 671 #endif 672 } 673 674 675 /* 676 * Polled input char. 677 */ 678 int 679 zs_getc(void *arg) 680 { 681 volatile struct zschan *zc = arg; 682 int s, c, rr0; 683 684 s = splhigh(); 685 /* Wait for a character to arrive. */ 686 do { 687 rr0 = zc->zc_csr; 688 ZS_DELAY(); 689 } while ((rr0 & ZSRR0_RX_READY) == 0); 690 691 c = zc->zc_data; 692 ZS_DELAY(); 693 splx(s); 694 695 /* 696 * This is used by the kd driver to read scan codes, 697 * so don't translate '\r' ==> '\n' here... 698 */ 699 return (c); 700 } 701 702 /* 703 * Polled output char. 704 */ 705 void 706 zs_putc(void *arg, int c) 707 { 708 volatile struct zschan *zc = arg; 709 int s, rr0; 710 711 s = splhigh(); 712 713 /* Wait for transmitter to become ready. */ 714 do { 715 rr0 = zc->zc_csr; 716 ZS_DELAY(); 717 } while ((rr0 & ZSRR0_TX_READY) == 0); 718 719 /* 720 * Send the next character. 721 * Now you'd think that this could be followed by a ZS_DELAY() 722 * just like all the other chip accesses, but it turns out that 723 * the `transmit-ready' interrupt isn't de-asserted until 724 * some period of time after the register write completes 725 * (more than a couple instructions). So to avoid stray 726 * interrupts we put in the 2us delay regardless of CPU model. 727 */ 728 zc->zc_data = c; 729 delay(2); 730 731 splx(s); 732 } 733 734 /*****************************************************************/ 735 736 737 738 739 /* 740 * Polled console input putchar. 741 */ 742 static int 743 zscngetc(dev_t dev) 744 { 745 return (zs_getc(zs_conschan_get)); 746 } 747 748 /* 749 * Polled console output putchar. 750 */ 751 static void 752 zscnputc(dev_t dev, int c) 753 { 754 zs_putc(zs_conschan_put, c); 755 } 756 757 int swallow_zsintrs; 758 759 static void 760 zscnpollc(dev_t dev, int on) 761 { 762 /* 763 * Need to tell zs driver to acknowledge all interrupts or we get 764 * annoying spurious interrupt messages. This is because mucking 765 * with spl() levels during polling does not prevent interrupts from 766 * being generated. 767 */ 768 769 if (on) swallow_zsintrs++; 770 else swallow_zsintrs--; 771 } 772 773 int 774 zs_console_flags(int promunit, int node, int channel) 775 { 776 int cookie, flags = 0; 777 char buf[255]; 778 779 /* 780 * We'll just do the OBP grovelling down here since that's 781 * the only type of firmware we support. 782 */ 783 784 /* Default to channel 0 if there are no explicit prom args */ 785 cookie = 0; 786 if (node == prom_instance_to_package(prom_stdin())) { 787 if (prom_getoption("input-device", buf, sizeof buf) == 0 && 788 strcmp("ttyb", buf) == 0) 789 cookie = 1; 790 791 if (channel == cookie) 792 flags |= ZS_HWFLAG_CONSOLE_INPUT; 793 } 794 795 if (node == prom_instance_to_package(prom_stdout())) { 796 if (prom_getoption("output-device", buf, sizeof buf) == 0 && 797 strcmp("ttyb", buf) == 0) 798 cookie = 1; 799 800 if (channel == cookie) 801 flags |= ZS_HWFLAG_CONSOLE_OUTPUT; 802 } 803 804 return (flags); 805 } 806 807