1 /* MN10300 On-chip serial driver for gdbstub I/O
 
   3  * Copyright (C) 2007 Red Hat, Inc. All Rights Reserved.
 
   4  * Written by David Howells (dhowells@redhat.com)
 
   6  * This program is free software; you can redistribute it and/or
 
   7  * modify it under the terms of the GNU General Public Licence
 
   8  * as published by the Free Software Foundation; either version
 
   9  * 2 of the Licence, or (at your option) any later version.
 
  11 #include <linux/string.h>
 
  12 #include <linux/kernel.h>
 
  13 #include <linux/signal.h>
 
  14 #include <linux/sched.h>
 
  16 #include <linux/console.h>
 
  17 #include <linux/init.h>
 
  18 #include <linux/tty.h>
 
  19 #include <asm/pgtable.h>
 
  20 #include <asm/system.h>
 
  21 #include <asm/gdb-stub.h>
 
  22 #include <asm/exceptions.h>
 
  23 #include <asm/unit/clock.h>
 
  24 #include "mn10300-serial.h"
 
  26 #if defined(CONFIG_GDBSTUB_ON_TTYSM0)
 
  27 struct mn10300_serial_port *const gdbstub_port = &mn10300_serial_port_sif0;
 
  28 #elif defined(CONFIG_GDBSTUB_ON_TTYSM1)
 
  29 struct mn10300_serial_port *const gdbstub_port = &mn10300_serial_port_sif1;
 
  31 struct mn10300_serial_port *const gdbstub_port = &mn10300_serial_port_sif2;
 
  36  * initialise the GDB stub I/O routines
 
  38 void __init gdbstub_io_init(void)
 
  43         switch (gdbstub_port->clock_src) {
 
  44         case MNSCx_CLOCK_SRC_IOCLK:
 
  45                 gdbstub_port->ioclk = MN10300_IOCLK;
 
  49         case MNSCx_CLOCK_SRC_IOBCLK:
 
  50                 gdbstub_port->ioclk = MN10300_IOBCLK;
 
  57         /* set up the serial port */
 
  58         gdbstub_io_set_baud(115200);
 
  60         /* we want to get serial receive interrupts */
 
  61         set_intr_level(gdbstub_port->rx_irq, GxICR_LEVEL_0);
 
  62         set_intr_level(gdbstub_port->tx_irq, GxICR_LEVEL_0);
 
  63         set_intr_stub(EXCEP_IRQ_LEVEL0, gdbstub_io_rx_handler);
 
  65         *gdbstub_port->rx_icr |= GxICR_ENABLE;
 
  66         tmp = *gdbstub_port->rx_icr;
 
  68         /* enable the device */
 
  69         scxctr = SC01CTR_CLN_8BIT;      /* 1N8 */
 
  70         switch (gdbstub_port->div_timer) {
 
  71         case MNSCx_DIV_TIMER_16BIT:
 
  72                 scxctr |= SC0CTR_CK_TM8UFLOW_8; /* == SC1CTR_CK_TM9UFLOW_8
 
  73                                                    == SC2CTR_CK_TM10UFLOW_8 */
 
  76         case MNSCx_DIV_TIMER_8BIT:
 
  77                 scxctr |= SC0CTR_CK_TM2UFLOW_8;
 
  81         scxctr |= SC01CTR_TXE | SC01CTR_RXE;
 
  83         *gdbstub_port->_control = scxctr;
 
  84         tmp = *gdbstub_port->_control;
 
  86         /* permit level 0 IRQs only */
 
  91                 : "i"(~EPSW_IM), "i"(EPSW_IE|EPSW_IM_1)
 
  96  * set up the GDB stub serial port baud rate timers
 
  98 void gdbstub_io_set_baud(unsigned baud)
 
 100         const unsigned bits = 10; /* 1 [start] + 8 [data] + 0 [parity] +
 
 102         unsigned long ioclk = gdbstub_port->ioclk;
 
 109         } else if (baud == 134) {
 
 110                 baud = 269;     /* 134 is really 134.5 */
 
 117         switch (gdbstub_port->div_timer) {
 
 118         case MNSCx_DIV_TIMER_16BIT:
 
 119                 tmxmd = TM8MD_SRC_IOCLK;
 
 120                 tmxbr = tmp = (ioclk / (baud * xdiv) + 4) / 8 - 1;
 
 121                 if (tmp > 0 && tmp <= 65535)
 
 124                 tmxmd = TM8MD_SRC_IOCLK_8;
 
 125                 tmxbr = tmp = (ioclk / (baud * 8 * xdiv) + 4) / 8 - 1;
 
 126                 if (tmp > 0 && tmp <= 65535)
 
 129                 tmxmd = TM8MD_SRC_IOCLK_32;
 
 130                 tmxbr = tmp = (ioclk / (baud * 32 * xdiv) + 4) / 8 - 1;
 
 131                 if (tmp > 0 && tmp <= 65535)
 
 136         case MNSCx_DIV_TIMER_8BIT:
 
 137                 tmxmd = TM2MD_SRC_IOCLK;
 
 138                 tmxbr = tmp = (ioclk / (baud * xdiv) + 4) / 8 - 1;
 
 139                 if (tmp > 0 && tmp <= 255)
 
 142                 tmxmd = TM2MD_SRC_IOCLK_8;
 
 143                 tmxbr = tmp = (ioclk / (baud * 8 * xdiv) + 4) / 8 - 1;
 
 144                 if (tmp > 0 && tmp <= 255)
 
 147                 tmxmd = TM2MD_SRC_IOCLK_32;
 
 148                 tmxbr = tmp = (ioclk / (baud * 32 * xdiv) + 4) / 8 - 1;
 
 149                 if (tmp > 0 && tmp <= 255)
 
 154         /* as a last resort, if the quotient is zero, default to 9600 bps */
 
 156         goto try_alternative;
 
 159         gdbstub_port->uart.timeout = (2 * bits * HZ) / baud;
 
 160         gdbstub_port->uart.timeout += HZ / 50;
 
 162         /* set the timer to produce the required baud rate */
 
 163         switch (gdbstub_port->div_timer) {
 
 164         case MNSCx_DIV_TIMER_16BIT:
 
 165                 *gdbstub_port->_tmxmd = 0;
 
 166                 *gdbstub_port->_tmxbr = tmxbr;
 
 167                 *gdbstub_port->_tmxmd = TM8MD_INIT_COUNTER;
 
 168                 *gdbstub_port->_tmxmd = tmxmd | TM8MD_COUNT_ENABLE;
 
 171         case MNSCx_DIV_TIMER_8BIT:
 
 172                 *gdbstub_port->_tmxmd = 0;
 
 173                 *(volatile u8 *) gdbstub_port->_tmxbr = (u8)tmxbr;
 
 174                 *gdbstub_port->_tmxmd = TM2MD_INIT_COUNTER;
 
 175                 *gdbstub_port->_tmxmd = tmxmd | TM2MD_COUNT_ENABLE;
 
 181  * wait for a character to come from the debugger
 
 183 int gdbstub_io_rx_char(unsigned char *_ch, int nonblock)
 
 190         if (gdbstub_rx_unget) {
 
 191                 *_ch = gdbstub_rx_unget;
 
 192                 gdbstub_rx_unget = 0;
 
 197         /* pull chars out of the buffer */
 
 198         ix = gdbstub_rx_outp;
 
 200         if (ix == gdbstub_rx_inp) {
 
 203 #ifdef CONFIG_MN10300_WD_TIMER
 
 204                 watchdog_alert_counter = 0;
 
 205 #endif /* CONFIG_MN10300_WD_TIMER */
 
 209         ch = gdbstub_rx_buffer[ix++];
 
 210         st = gdbstub_rx_buffer[ix++];
 
 212         gdbstub_rx_outp = ix & (PAGE_SIZE - 1);
 
 214         st &= SC01STR_RXF | SC01STR_RBF | SC01STR_FEF | SC01STR_PEF |
 
 217         /* deal with what we've got
 
 218          * - note that the UART doesn't do BREAK-detection for us
 
 220         if (st & SC01STR_FEF && ch == 0) {
 
 221                 switch (gdbstub_port->rx_brk) {
 
 222                 case 0: gdbstub_port->rx_brk = 1;       goto try_again;
 
 223                 case 1: gdbstub_port->rx_brk = 2;       goto try_again;
 
 225                         gdbstub_port->rx_brk = 3;
 
 226                         gdbstub_proto("### GDB MNSERIAL Rx Break Detected"
 
 232         } else if (st & SC01STR_FEF) {
 
 233                 if (gdbstub_port->rx_brk)
 
 236                 gdbstub_proto("### GDB MNSERIAL Framing Error ###\n");
 
 238         } else if (st & SC01STR_OEF) {
 
 239                 if (gdbstub_port->rx_brk)
 
 242                 gdbstub_proto("### GDB MNSERIAL Overrun Error ###\n");
 
 244         } else if (st & SC01STR_PEF) {
 
 245                 if (gdbstub_port->rx_brk)
 
 248                 gdbstub_proto("### GDB MNSERIAL Parity Error ###\n");
 
 251                 /* look for the tail-end char on a break run */
 
 252                 if (gdbstub_port->rx_brk == 3) {
 
 263                                 gdbstub_port->rx_brk = 0;
 
 270                 gdbstub_port->rx_brk = 0;
 
 271                 gdbstub_io("### GDB Rx %02x (st=%02x) ###\n", ch, st);
 
 278  * send a character to the debugger
 
 280 void gdbstub_io_tx_char(unsigned char ch)
 
 282         while (*gdbstub_port->_status & SC01STR_TBF)
 
 286                 *(u8 *) gdbstub_port->_txb = 0x0d;
 
 287                 while (*gdbstub_port->_status & SC01STR_TBF)
 
 291         *(u8 *) gdbstub_port->_txb = ch;
 
 295  * flush the transmission buffers
 
 297 void gdbstub_io_tx_flush(void)
 
 299         while (*gdbstub_port->_status & (SC01STR_TBF | SC01STR_TXF))