| /* 16550 serial driver for gdbstub I/O |
| * |
| * Copyright (C) 2007 Red Hat, Inc. All Rights Reserved. |
| * Written by David Howells (dhowells@redhat.com) |
| * |
| * This program is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU General Public Licence |
| * as published by the Free Software Foundation; either version |
| * 2 of the Licence, or (at your option) any later version. |
| */ |
| #include <linux/string.h> |
| #include <linux/kernel.h> |
| #include <linux/signal.h> |
| #include <linux/sched.h> |
| #include <linux/mm.h> |
| #include <linux/console.h> |
| #include <linux/init.h> |
| #include <linux/nmi.h> |
| |
| #include <asm/pgtable.h> |
| #include <asm/gdb-stub.h> |
| #include <asm/exceptions.h> |
| #include <asm/serial-regs.h> |
| #include <unit/serial.h> |
| #include <asm/smp.h> |
| |
| /* |
| * initialise the GDB stub |
| */ |
| void gdbstub_io_init(void) |
| { |
| u16 tmp; |
| |
| /* set up the serial port */ |
| GDBPORT_SERIAL_LCR = UART_LCR_WLEN8; /* 1N8 */ |
| GDBPORT_SERIAL_FCR = (UART_FCR_ENABLE_FIFO | UART_FCR_CLEAR_RCVR | |
| UART_FCR_CLEAR_XMIT); |
| |
| FLOWCTL_CLEAR(DTR); |
| FLOWCTL_SET(RTS); |
| |
| gdbstub_io_set_baud(115200); |
| |
| /* we want to get serial receive interrupts */ |
| XIRQxICR(GDBPORT_SERIAL_IRQ) = 0; |
| tmp = XIRQxICR(GDBPORT_SERIAL_IRQ); |
| |
| #if CONFIG_GDBSTUB_IRQ_LEVEL == 0 |
| IVAR0 = EXCEP_IRQ_LEVEL0; |
| #elif CONFIG_GDBSTUB_IRQ_LEVEL == 1 |
| IVAR1 = EXCEP_IRQ_LEVEL1; |
| #elif CONFIG_GDBSTUB_IRQ_LEVEL == 2 |
| IVAR2 = EXCEP_IRQ_LEVEL2; |
| #elif CONFIG_GDBSTUB_IRQ_LEVEL == 3 |
| IVAR3 = EXCEP_IRQ_LEVEL3; |
| #elif CONFIG_GDBSTUB_IRQ_LEVEL == 4 |
| IVAR4 = EXCEP_IRQ_LEVEL4; |
| #elif CONFIG_GDBSTUB_IRQ_LEVEL == 5 |
| IVAR5 = EXCEP_IRQ_LEVEL5; |
| #else |
| #error "Unknown irq level for gdbstub." |
| #endif |
| |
| set_intr_stub(NUM2EXCEP_IRQ_LEVEL(CONFIG_GDBSTUB_IRQ_LEVEL), |
| gdbstub_io_rx_handler); |
| |
| XIRQxICR(GDBPORT_SERIAL_IRQ) &= ~GxICR_REQUEST; |
| XIRQxICR(GDBPORT_SERIAL_IRQ) = |
| GxICR_ENABLE | NUM2GxICR_LEVEL(CONFIG_GDBSTUB_IRQ_LEVEL); |
| tmp = XIRQxICR(GDBPORT_SERIAL_IRQ); |
| |
| GDBPORT_SERIAL_IER = UART_IER_RDI | UART_IER_RLSI; |
| |
| /* permit level 0 IRQs to take place */ |
| arch_local_change_intr_mask_level( |
| NUM2EPSW_IM(CONFIG_GDBSTUB_IRQ_LEVEL + 1)); |
| } |
| |
| /* |
| * set up the GDB stub serial port baud rate timers |
| */ |
| void gdbstub_io_set_baud(unsigned baud) |
| { |
| unsigned value; |
| u8 lcr; |
| |
| value = 18432000 / 16 / baud; |
| |
| lcr = GDBPORT_SERIAL_LCR; |
| GDBPORT_SERIAL_LCR |= UART_LCR_DLAB; |
| GDBPORT_SERIAL_DLL = value & 0xff; |
| GDBPORT_SERIAL_DLM = (value >> 8) & 0xff; |
| GDBPORT_SERIAL_LCR = lcr; |
| } |
| |
| /* |
| * wait for a character to come from the debugger |
| */ |
| int gdbstub_io_rx_char(unsigned char *_ch, int nonblock) |
| { |
| unsigned ix; |
| u8 ch, st; |
| #if defined(CONFIG_MN10300_WD_TIMER) |
| int cpu; |
| #endif |
| |
| *_ch = 0xff; |
| |
| if (gdbstub_rx_unget) { |
| *_ch = gdbstub_rx_unget; |
| gdbstub_rx_unget = 0; |
| return 0; |
| } |
| |
| try_again: |
| /* pull chars out of the buffer */ |
| ix = gdbstub_rx_outp; |
| barrier(); |
| if (ix == gdbstub_rx_inp) { |
| if (nonblock) |
| return -EAGAIN; |
| #ifdef CONFIG_MN10300_WD_TIMER |
| for (cpu = 0; cpu < NR_CPUS; cpu++) |
| watchdog_alert_counter[cpu] = 0; |
| #endif |
| goto try_again; |
| } |
| |
| ch = gdbstub_rx_buffer[ix++]; |
| st = gdbstub_rx_buffer[ix++]; |
| barrier(); |
| gdbstub_rx_outp = ix & 0x00000fff; |
| |
| if (st & UART_LSR_BI) { |
| gdbstub_proto("### GDB Rx Break Detected ###\n"); |
| return -EINTR; |
| } else if (st & (UART_LSR_FE | UART_LSR_OE | UART_LSR_PE)) { |
| gdbstub_proto("### GDB Rx Error (st=%02x) ###\n", st); |
| return -EIO; |
| } else { |
| gdbstub_proto("### GDB Rx %02x (st=%02x) ###\n", ch, st); |
| *_ch = ch & 0x7f; |
| return 0; |
| } |
| } |
| |
| /* |
| * send a character to the debugger |
| */ |
| void gdbstub_io_tx_char(unsigned char ch) |
| { |
| FLOWCTL_SET(DTR); |
| LSR_WAIT_FOR(THRE); |
| /* FLOWCTL_WAIT_FOR(CTS); */ |
| |
| if (ch == 0x0a) { |
| GDBPORT_SERIAL_TX = 0x0d; |
| LSR_WAIT_FOR(THRE); |
| /* FLOWCTL_WAIT_FOR(CTS); */ |
| } |
| GDBPORT_SERIAL_TX = ch; |
| |
| FLOWCTL_CLEAR(DTR); |
| } |
| |
| /* |
| * send a character to the debugger |
| */ |
| void gdbstub_io_tx_flush(void) |
| { |
| LSR_WAIT_FOR(TEMT); |
| LSR_WAIT_FOR(THRE); |
| FLOWCTL_CLEAR(DTR); |
| } |