1 1.5 tsutsui /* $NetBSD: ns16550.c,v 1.5 2022/09/05 14:14:42 tsutsui Exp $ */ 2 1.1 thorpej 3 1.1 thorpej /* 4 1.1 thorpej * Copyright (c) 2002 Wasabi Systems, Inc. 5 1.1 thorpej * All rights reserved. 6 1.1 thorpej * 7 1.1 thorpej * Written by Jason R. Thorpe for Wasabi Systems, Inc. 8 1.1 thorpej * 9 1.1 thorpej * Redistribution and use in source and binary forms, with or without 10 1.1 thorpej * modification, are permitted provided that the following conditions 11 1.1 thorpej * are met: 12 1.1 thorpej * 1. Redistributions of source code must retain the above copyright 13 1.1 thorpej * notice, this list of conditions and the following disclaimer. 14 1.1 thorpej * 2. Redistributions in binary form must reproduce the above copyright 15 1.1 thorpej * notice, this list of conditions and the following disclaimer in the 16 1.1 thorpej * documentation and/or other materials provided with the distribution. 17 1.1 thorpej * 3. All advertising materials mentioning features or use of this software 18 1.1 thorpej * must display the following acknowledgement: 19 1.1 thorpej * This product includes software developed for the NetBSD Project by 20 1.1 thorpej * Wasabi Systems, Inc. 21 1.1 thorpej * 4. The name of Wasabi Systems, Inc. may not be used to endorse 22 1.1 thorpej * or promote products derived from this software without specific prior 23 1.1 thorpej * written permission. 24 1.1 thorpej * 25 1.1 thorpej * THIS SOFTWARE IS PROVIDED BY WASABI SYSTEMS, INC. ``AS IS'' AND 26 1.1 thorpej * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 27 1.1 thorpej * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 28 1.1 thorpej * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL WASABI SYSTEMS, INC 29 1.1 thorpej * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 30 1.1 thorpej * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 31 1.1 thorpej * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 32 1.1 thorpej * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 33 1.1 thorpej * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 34 1.1 thorpej * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 35 1.1 thorpej * POSSIBILITY OF SUCH DAMAGE. 36 1.1 thorpej */ 37 1.1 thorpej 38 1.1 thorpej /* 39 1.1 thorpej * This file provides the cons_init() function and console I/O routines 40 1.1 thorpej * for boards that use 16550-compatible UARTs. 41 1.1 thorpej */ 42 1.1 thorpej 43 1.1 thorpej #include <sys/types.h> 44 1.1 thorpej #include <dev/ic/comreg.h> 45 1.1 thorpej #include <lib/libsa/stand.h> 46 1.1 thorpej 47 1.1 thorpej #include "board.h" 48 1.1 thorpej 49 1.4 cliff #ifdef NS16550_AX4 50 1.4 cliff #define INB(x) *((volatile uint8_t *) (CONADDR + ((x) * 4))) 51 1.4 cliff #define OUTB(x, v) *((volatile uint8_t *) (CONADDR + ((x) * 4))) = (v) 52 1.4 cliff #else 53 1.3 perry #define INB(x) *((volatile uint8_t *) (CONADDR + (x))) 54 1.3 perry #define OUTB(x, v) *((volatile uint8_t *) (CONADDR + (x))) = (v) 55 1.4 cliff #endif 56 1.1 thorpej 57 1.2 thorpej #ifndef NS16550_FREQ 58 1.2 thorpej #define NS16550_FREQ COM_FREQ 59 1.2 thorpej #endif 60 1.2 thorpej 61 1.1 thorpej static int 62 1.1 thorpej comspeed(int speed) 63 1.1 thorpej { 64 1.1 thorpej #define divrnd(n, q) (((n)*2/(q)+1)/2) /* divide and round off */ 65 1.1 thorpej 66 1.1 thorpej int x, err; 67 1.1 thorpej 68 1.1 thorpej if (speed <= 0) 69 1.1 thorpej return (-1); 70 1.2 thorpej x = divrnd((NS16550_FREQ / 16), speed); 71 1.1 thorpej if (x <= 0) 72 1.1 thorpej return (-1); 73 1.2 thorpej err = divrnd((((quad_t)NS16550_FREQ) / 16) * 1000, speed * x) - 1000; 74 1.1 thorpej if (err < 0) 75 1.1 thorpej err = -err; 76 1.1 thorpej if (err > COM_TOLERANCE) 77 1.1 thorpej return (-1); 78 1.1 thorpej return (x); 79 1.1 thorpej #undef divrnd 80 1.1 thorpej } 81 1.1 thorpej 82 1.1 thorpej void 83 1.1 thorpej cons_init(void) 84 1.1 thorpej { 85 1.1 thorpej int rate; 86 1.1 thorpej 87 1.1 thorpej OUTB(com_cfcr, LCR_DLAB); 88 1.1 thorpej rate = comspeed(CONSPEED); 89 1.1 thorpej OUTB(com_dlbl, rate); 90 1.1 thorpej OUTB(com_dlbh, rate >> 8); 91 1.1 thorpej OUTB(com_cfcr, LCR_8BITS); 92 1.1 thorpej OUTB(com_mcr, MCR_DTR | MCR_RTS); 93 1.1 thorpej OUTB(com_fifo, 94 1.1 thorpej FIFO_ENABLE | FIFO_RCV_RST | FIFO_XMT_RST | FIFO_TRIGGER_1); 95 1.1 thorpej OUTB(com_ier, 0); 96 1.1 thorpej } 97 1.1 thorpej 98 1.1 thorpej int 99 1.1 thorpej getchar(void) 100 1.1 thorpej { 101 1.1 thorpej uint8_t stat; 102 1.1 thorpej 103 1.1 thorpej while (!ISSET(stat = INB(com_lsr), LSR_RXRDY)) 104 1.1 thorpej /* spin */ ; 105 1.1 thorpej return (INB(com_data)); 106 1.1 thorpej } 107 1.1 thorpej 108 1.1 thorpej static void 109 1.1 thorpej iputchar(int c) 110 1.1 thorpej { 111 1.1 thorpej uint8_t stat; 112 1.1 thorpej int timo; 113 1.1 thorpej 114 1.1 thorpej /* Wait for any pending transmission to finish. */ 115 1.1 thorpej timo = 50000; 116 1.1 thorpej while (!ISSET(stat = INB(com_lsr), LSR_TXRDY) && --timo) 117 1.1 thorpej /* spin */ ; 118 1.1 thorpej 119 1.1 thorpej OUTB(com_data, c); 120 1.1 thorpej 121 1.1 thorpej /* Wait for this transmission to complete. */ 122 1.1 thorpej timo = 1500000; 123 1.1 thorpej while (!ISSET(stat = INB(com_lsr), LSR_TXRDY) && --timo) 124 1.1 thorpej /* spin */ ; 125 1.1 thorpej 126 1.1 thorpej /* Clear any interrupts generated by this transmission. */ 127 1.1 thorpej (void) INB(com_iir); 128 1.1 thorpej } 129 1.1 thorpej 130 1.1 thorpej void 131 1.1 thorpej putchar(int c) 132 1.1 thorpej { 133 1.1 thorpej 134 1.1 thorpej if (c == '\n') 135 1.1 thorpej iputchar('\r'); 136 1.1 thorpej iputchar(c); 137 1.1 thorpej } 138