mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-10 15:19:51 +00:00
serial: sh-sci: Implement CONSOLE_POLL support and kill off old kgdb console.
Signed-off-by: Paul Mundt <lethal@linux-sh.org>
This commit is contained in:
parent
ab6e570ba3
commit
07d2a1a1cd
@ -51,7 +51,6 @@
|
||||
#ifdef CONFIG_SUPERH
|
||||
#include <asm/clock.h>
|
||||
#include <asm/sh_bios.h>
|
||||
#include <asm/kgdb.h>
|
||||
#endif
|
||||
|
||||
#include "sh-sci.h"
|
||||
@ -85,10 +84,6 @@ struct sci_port {
|
||||
#endif
|
||||
};
|
||||
|
||||
#ifdef CONFIG_SH_KGDB
|
||||
static struct sci_port *kgdb_sci_port;
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_SERIAL_SH_SCI_CONSOLE
|
||||
static struct sci_port *serial_console_port;
|
||||
#endif
|
||||
@ -107,21 +102,18 @@ to_sci_port(struct uart_port *uart)
|
||||
return container_of(uart, struct sci_port, port);
|
||||
}
|
||||
|
||||
#if defined(CONFIG_SERIAL_SH_SCI_CONSOLE) && \
|
||||
defined(CONFIG_SH_STANDARD_BIOS) || defined(CONFIG_SH_KGDB)
|
||||
#if defined(CONFIG_CONSOLE_POLL) || defined(CONFIG_SERIAL_SH_SCI_CONSOLE)
|
||||
static inline void handle_error(struct uart_port *port)
|
||||
{
|
||||
/* Clear error flags */
|
||||
sci_out(port, SCxSR, SCxSR_ERROR_CLEAR(port));
|
||||
}
|
||||
|
||||
static int get_char(struct uart_port *port)
|
||||
static int sci_poll_get_char(struct uart_port *port)
|
||||
{
|
||||
unsigned long flags;
|
||||
unsigned short status;
|
||||
int c;
|
||||
|
||||
spin_lock_irqsave(&port->lock, flags);
|
||||
do {
|
||||
status = sci_in(port, SCxSR);
|
||||
if (status & SCxSR_ERRORS(port)) {
|
||||
@ -129,24 +121,20 @@ static int get_char(struct uart_port *port)
|
||||
continue;
|
||||
}
|
||||
} while (!(status & SCxSR_RDxF(port)));
|
||||
|
||||
c = sci_in(port, SCxRDR);
|
||||
|
||||
/* Dummy read */
|
||||
sci_in(port, SCxSR);
|
||||
sci_out(port, SCxSR, SCxSR_RDxF_CLEAR(port));
|
||||
spin_unlock_irqrestore(&port->lock, flags);
|
||||
|
||||
return c;
|
||||
}
|
||||
#endif /* CONFIG_SH_STANDARD_BIOS || CONFIG_SH_KGDB */
|
||||
|
||||
#if defined(CONFIG_SERIAL_SH_SCI_CONSOLE) || defined(CONFIG_SH_KGDB)
|
||||
static void put_char(struct uart_port *port, char c)
|
||||
static void sci_poll_put_char(struct uart_port *port, unsigned char c)
|
||||
{
|
||||
unsigned long flags;
|
||||
unsigned short status;
|
||||
|
||||
spin_lock_irqsave(&port->lock, flags);
|
||||
|
||||
do {
|
||||
status = sci_in(port, SCxSR);
|
||||
} while (!(status & SCxSR_TDxE(port)));
|
||||
@ -154,82 +142,8 @@ static void put_char(struct uart_port *port, char c)
|
||||
sci_in(port, SCxSR); /* Dummy read */
|
||||
sci_out(port, SCxSR, SCxSR_TDxE_CLEAR(port));
|
||||
sci_out(port, SCxTDR, c);
|
||||
|
||||
spin_unlock_irqrestore(&port->lock, flags);
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_SERIAL_SH_SCI_CONSOLE
|
||||
static void put_string(struct sci_port *sci_port, const char *buffer, int count)
|
||||
{
|
||||
struct uart_port *port = &sci_port->port;
|
||||
const unsigned char *p = buffer;
|
||||
int i;
|
||||
|
||||
#if defined(CONFIG_SH_STANDARD_BIOS) || defined(CONFIG_SH_KGDB)
|
||||
int checksum;
|
||||
int usegdb = 0;
|
||||
|
||||
#ifdef CONFIG_SH_STANDARD_BIOS
|
||||
/* This call only does a trap the first time it is
|
||||
* called, and so is safe to do here unconditionally
|
||||
*/
|
||||
usegdb |= sh_bios_in_gdb_mode();
|
||||
#endif
|
||||
#ifdef CONFIG_SH_KGDB
|
||||
usegdb |= (kgdb_in_gdb_mode && (sci_port == kgdb_sci_port));
|
||||
#endif
|
||||
|
||||
if (usegdb) {
|
||||
/* $<packet info>#<checksum>. */
|
||||
do {
|
||||
unsigned char c;
|
||||
put_char(port, '$');
|
||||
put_char(port, 'O'); /* 'O'utput to console */
|
||||
checksum = 'O';
|
||||
|
||||
/* Don't use run length encoding */
|
||||
for (i = 0; i < count; i++) {
|
||||
int h, l;
|
||||
|
||||
c = *p++;
|
||||
h = hex_asc_hi(c);
|
||||
l = hex_asc_lo(c);
|
||||
put_char(port, h);
|
||||
put_char(port, l);
|
||||
checksum += h + l;
|
||||
}
|
||||
put_char(port, '#');
|
||||
put_char(port, hex_asc_hi(checksum));
|
||||
put_char(port, hex_asc_lo(checksum));
|
||||
} while (get_char(port) != '+');
|
||||
} else
|
||||
#endif /* CONFIG_SH_STANDARD_BIOS || CONFIG_SH_KGDB */
|
||||
for (i = 0; i < count; i++) {
|
||||
if (*p == 10)
|
||||
put_char(port, '\r');
|
||||
put_char(port, *p++);
|
||||
}
|
||||
}
|
||||
#endif /* CONFIG_SERIAL_SH_SCI_CONSOLE */
|
||||
|
||||
#ifdef CONFIG_SH_KGDB
|
||||
static int kgdb_sci_getchar(void)
|
||||
{
|
||||
int c;
|
||||
|
||||
/* Keep trying to read a character, this could be neater */
|
||||
while ((c = get_char(&kgdb_sci_port->port)) < 0)
|
||||
cpu_relax();
|
||||
|
||||
return c;
|
||||
}
|
||||
|
||||
static inline void kgdb_sci_putchar(int c)
|
||||
{
|
||||
put_char(&kgdb_sci_port->port, c);
|
||||
}
|
||||
#endif /* CONFIG_SH_KGDB */
|
||||
#endif /* CONFIG_CONSOLE_POLL || CONFIG_SERIAL_SH_SCI_CONSOLE */
|
||||
|
||||
#if defined(__H8300S__)
|
||||
enum { sci_disable, sci_enable };
|
||||
@ -1181,6 +1095,10 @@ static struct uart_ops sci_uart_ops = {
|
||||
.request_port = sci_request_port,
|
||||
.config_port = sci_config_port,
|
||||
.verify_port = sci_verify_port,
|
||||
#ifdef CONFIG_CONSOLE_POLL
|
||||
.poll_get_char = sci_poll_get_char,
|
||||
.poll_put_char = sci_poll_put_char,
|
||||
#endif
|
||||
};
|
||||
|
||||
static void __init sci_init_ports(void)
|
||||
@ -1247,7 +1165,15 @@ int __init early_sci_setup(struct uart_port *port)
|
||||
static void serial_console_write(struct console *co, const char *s,
|
||||
unsigned count)
|
||||
{
|
||||
put_string(serial_console_port, s, count);
|
||||
struct uart_port *port = &serial_console_port->port;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < count; i++) {
|
||||
if (*s == 10)
|
||||
sci_poll_put_char(port, '\r');
|
||||
|
||||
sci_poll_put_char(port, *s++);
|
||||
}
|
||||
}
|
||||
|
||||
static int __init serial_console_setup(struct console *co, char *options)
|
||||
@ -1325,88 +1251,7 @@ static int __init sci_console_init(void)
|
||||
console_initcall(sci_console_init);
|
||||
#endif /* CONFIG_SERIAL_SH_SCI_CONSOLE */
|
||||
|
||||
#ifdef CONFIG_SH_KGDB_CONSOLE
|
||||
/*
|
||||
* FIXME: Most of this can go away.. at the moment, we rely on
|
||||
* arch/sh/kernel/setup.c to do the command line parsing for kgdb, though
|
||||
* most of that can easily be done here instead.
|
||||
*
|
||||
* For the time being, just accept the values that were parsed earlier..
|
||||
*/
|
||||
static void __init kgdb_console_get_options(struct uart_port *port, int *baud,
|
||||
int *parity, int *bits)
|
||||
{
|
||||
*baud = kgdb_baud;
|
||||
*parity = tolower(kgdb_parity);
|
||||
*bits = kgdb_bits - '0';
|
||||
}
|
||||
|
||||
/*
|
||||
* The naming here is somewhat misleading, since kgdb_console_setup() takes
|
||||
* care of the early-on initialization for kgdb, regardless of whether we
|
||||
* actually use kgdb as a console or not.
|
||||
*
|
||||
* On the plus side, this lets us kill off the old kgdb_sci_setup() nonsense.
|
||||
*/
|
||||
int __init kgdb_console_setup(struct console *co, char *options)
|
||||
{
|
||||
struct uart_port *port = &sci_ports[kgdb_portnum].port;
|
||||
int baud = 38400;
|
||||
int bits = 8;
|
||||
int parity = 'n';
|
||||
int flow = 'n';
|
||||
|
||||
if (co->index != kgdb_portnum)
|
||||
co->index = kgdb_portnum;
|
||||
|
||||
kgdb_sci_port = &sci_ports[co->index];
|
||||
port = &kgdb_sci_port->port;
|
||||
|
||||
/*
|
||||
* Also need to check port->type, we don't actually have any
|
||||
* UPIO_PORT ports, but uart_report_port() handily misreports
|
||||
* it anyways if we don't have a port available by the time this is
|
||||
* called.
|
||||
*/
|
||||
if (!port->type)
|
||||
return -ENODEV;
|
||||
if (!port->membase || !port->mapbase)
|
||||
return -ENODEV;
|
||||
|
||||
if (options)
|
||||
uart_parse_options(options, &baud, &parity, &bits, &flow);
|
||||
else
|
||||
kgdb_console_get_options(port, &baud, &parity, &bits);
|
||||
|
||||
kgdb_getchar = kgdb_sci_getchar;
|
||||
kgdb_putchar = kgdb_sci_putchar;
|
||||
|
||||
return uart_set_options(port, co, baud, parity, bits, flow);
|
||||
}
|
||||
|
||||
static struct console kgdb_console = {
|
||||
.name = "ttySC",
|
||||
.device = uart_console_device,
|
||||
.write = kgdb_console_write,
|
||||
.setup = kgdb_console_setup,
|
||||
.flags = CON_PRINTBUFFER,
|
||||
.index = -1,
|
||||
.data = &sci_uart_driver,
|
||||
};
|
||||
|
||||
/* Register the KGDB console so we get messages (d'oh!) */
|
||||
static int __init kgdb_console_init(void)
|
||||
{
|
||||
sci_init_ports();
|
||||
register_console(&kgdb_console);
|
||||
return 0;
|
||||
}
|
||||
console_initcall(kgdb_console_init);
|
||||
#endif /* CONFIG_SH_KGDB_CONSOLE */
|
||||
|
||||
#if defined(CONFIG_SH_KGDB_CONSOLE)
|
||||
#define SCI_CONSOLE (&kgdb_console)
|
||||
#elif defined(CONFIG_SERIAL_SH_SCI_CONSOLE)
|
||||
#if defined(CONFIG_SERIAL_SH_SCI_CONSOLE)
|
||||
#define SCI_CONSOLE (&serial_console)
|
||||
#else
|
||||
#define SCI_CONSOLE 0
|
||||
@ -1481,12 +1326,6 @@ static int __devinit sci_probe(struct platform_device *dev)
|
||||
uart_add_one_port(&sci_uart_driver, &sciport->port);
|
||||
}
|
||||
|
||||
#if defined(CONFIG_SH_KGDB) && !defined(CONFIG_SH_KGDB_CONSOLE)
|
||||
kgdb_sci_port = &sci_ports[kgdb_portnum];
|
||||
kgdb_getchar = kgdb_sci_getchar;
|
||||
kgdb_putchar = kgdb_sci_putchar;
|
||||
#endif
|
||||
|
||||
#if defined(CONFIG_CPU_FREQ) && defined(CONFIG_HAVE_CLK)
|
||||
cpufreq_register_notifier(&sci_nb, CPUFREQ_TRANSITION_NOTIFIER);
|
||||
dev_info(&dev->dev, "CPU frequency notifier registered\n");
|
||||
|
Loading…
x
Reference in New Issue
Block a user