xref: /openbmc/linux/drivers/tty/serial/8250/8250_rt288x.c (revision 2612e3bbc0386368a850140a6c9b990cd496a5ec)
1b334214eSIlpo Järvinen // SPDX-License-Identifier: GPL-2.0
2b334214eSIlpo Järvinen /*
3b334214eSIlpo Järvinen  * RT288x/Au1xxx driver
4b334214eSIlpo Järvinen  */
5b334214eSIlpo Järvinen 
6b334214eSIlpo Järvinen #include <linux/module.h>
7b334214eSIlpo Järvinen #include <linux/io.h>
8b334214eSIlpo Järvinen #include <linux/init.h>
9b334214eSIlpo Järvinen #include <linux/console.h>
10b334214eSIlpo Järvinen #include <linux/serial.h>
11b334214eSIlpo Järvinen #include <linux/serial_8250.h>
12b334214eSIlpo Järvinen 
13b334214eSIlpo Järvinen #include "8250.h"
14b334214eSIlpo Järvinen 
1533e3b0ebSIlpo Järvinen #define RT288X_DL	0x28
1633e3b0ebSIlpo Järvinen 
17b334214eSIlpo Järvinen /* Au1x00/RT288x UART hardware has a weird register layout */
18*72fc578bSIlpo Järvinen static const u8 au_io_in_map[7] = {
19b334214eSIlpo Järvinen 	[UART_RX]	= 0,
20b334214eSIlpo Järvinen 	[UART_IER]	= 2,
21b334214eSIlpo Järvinen 	[UART_IIR]	= 3,
22b334214eSIlpo Järvinen 	[UART_LCR]	= 5,
23b334214eSIlpo Järvinen 	[UART_MCR]	= 6,
24b334214eSIlpo Järvinen 	[UART_LSR]	= 7,
25b334214eSIlpo Järvinen 	[UART_MSR]	= 8,
26b334214eSIlpo Järvinen };
27b334214eSIlpo Järvinen 
28*72fc578bSIlpo Järvinen static const u8 au_io_out_map[5] = {
29b334214eSIlpo Järvinen 	[UART_TX]	= 1,
30b334214eSIlpo Järvinen 	[UART_IER]	= 2,
31b334214eSIlpo Järvinen 	[UART_FCR]	= 4,
32b334214eSIlpo Järvinen 	[UART_LCR]	= 5,
33b334214eSIlpo Järvinen 	[UART_MCR]	= 6,
34b334214eSIlpo Järvinen };
35b334214eSIlpo Järvinen 
au_serial_in(struct uart_port * p,int offset)36b334214eSIlpo Järvinen static unsigned int au_serial_in(struct uart_port *p, int offset)
37b334214eSIlpo Järvinen {
38b334214eSIlpo Järvinen 	if (offset >= ARRAY_SIZE(au_io_in_map))
39b334214eSIlpo Järvinen 		return UINT_MAX;
40b334214eSIlpo Järvinen 	offset = au_io_in_map[offset];
41*72fc578bSIlpo Järvinen 
42b334214eSIlpo Järvinen 	return __raw_readl(p->membase + (offset << p->regshift));
43b334214eSIlpo Järvinen }
44b334214eSIlpo Järvinen 
au_serial_out(struct uart_port * p,int offset,int value)45b334214eSIlpo Järvinen static void au_serial_out(struct uart_port *p, int offset, int value)
46b334214eSIlpo Järvinen {
47b334214eSIlpo Järvinen 	if (offset >= ARRAY_SIZE(au_io_out_map))
48b334214eSIlpo Järvinen 		return;
49b334214eSIlpo Järvinen 	offset = au_io_out_map[offset];
50*72fc578bSIlpo Järvinen 
51b334214eSIlpo Järvinen 	__raw_writel(value, p->membase + (offset << p->regshift));
52b334214eSIlpo Järvinen }
53b334214eSIlpo Järvinen 
54b334214eSIlpo Järvinen /* Au1x00 haven't got a standard divisor latch */
au_serial_dl_read(struct uart_8250_port * up)55b334214eSIlpo Järvinen static u32 au_serial_dl_read(struct uart_8250_port *up)
56b334214eSIlpo Järvinen {
5733e3b0ebSIlpo Järvinen 	return __raw_readl(up->port.membase + RT288X_DL);
58b334214eSIlpo Järvinen }
59b334214eSIlpo Järvinen 
au_serial_dl_write(struct uart_8250_port * up,u32 value)60b334214eSIlpo Järvinen static void au_serial_dl_write(struct uart_8250_port *up, u32 value)
61b334214eSIlpo Järvinen {
6233e3b0ebSIlpo Järvinen 	__raw_writel(value, up->port.membase + RT288X_DL);
63b334214eSIlpo Järvinen }
64b334214eSIlpo Järvinen 
au_platform_setup(struct plat_serial8250_port * p)65b334214eSIlpo Järvinen int au_platform_setup(struct plat_serial8250_port *p)
66b334214eSIlpo Järvinen {
67b334214eSIlpo Järvinen 	p->iotype = UPIO_AU;
68b334214eSIlpo Järvinen 
69b334214eSIlpo Järvinen 	p->serial_in = au_serial_in;
70b334214eSIlpo Järvinen 	p->serial_out = au_serial_out;
71b334214eSIlpo Järvinen 	p->dl_read = au_serial_dl_read;
72b334214eSIlpo Järvinen 	p->dl_write = au_serial_dl_write;
73b334214eSIlpo Järvinen 
74b334214eSIlpo Järvinen 	p->mapsize = 0x1000;
75b334214eSIlpo Järvinen 
76b334214eSIlpo Järvinen 	p->bugs |= UART_BUG_NOMSR;
77b334214eSIlpo Järvinen 
78b334214eSIlpo Järvinen 	return 0;
79b334214eSIlpo Järvinen }
80b334214eSIlpo Järvinen EXPORT_SYMBOL_GPL(au_platform_setup);
81b334214eSIlpo Järvinen 
rt288x_setup(struct uart_port * p)82b334214eSIlpo Järvinen int rt288x_setup(struct uart_port *p)
83b334214eSIlpo Järvinen {
84b334214eSIlpo Järvinen 	struct uart_8250_port *up = up_to_u8250p(p);
85b334214eSIlpo Järvinen 
86b334214eSIlpo Järvinen 	p->iotype = UPIO_AU;
87b334214eSIlpo Järvinen 
88b334214eSIlpo Järvinen 	p->serial_in = au_serial_in;
89b334214eSIlpo Järvinen 	p->serial_out = au_serial_out;
90b334214eSIlpo Järvinen 	up->dl_read = au_serial_dl_read;
91b334214eSIlpo Järvinen 	up->dl_write = au_serial_dl_write;
92b334214eSIlpo Järvinen 
93b334214eSIlpo Järvinen 	p->mapsize = 0x100;
94b334214eSIlpo Järvinen 
95b334214eSIlpo Järvinen 	up->bugs |= UART_BUG_NOMSR;
96b334214eSIlpo Järvinen 
97b334214eSIlpo Järvinen 	return 0;
98b334214eSIlpo Järvinen }
99b334214eSIlpo Järvinen EXPORT_SYMBOL_GPL(rt288x_setup);
100b334214eSIlpo Järvinen 
101b334214eSIlpo Järvinen #ifdef CONFIG_SERIAL_8250_CONSOLE
au_putc(struct uart_port * port,unsigned char c)102b334214eSIlpo Järvinen static void au_putc(struct uart_port *port, unsigned char c)
103b334214eSIlpo Järvinen {
104b334214eSIlpo Järvinen 	unsigned int status;
105b334214eSIlpo Järvinen 
106b334214eSIlpo Järvinen 	au_serial_out(port, UART_TX, c);
107b334214eSIlpo Järvinen 
108b334214eSIlpo Järvinen 	for (;;) {
109b334214eSIlpo Järvinen 		status = au_serial_in(port, UART_LSR);
110b334214eSIlpo Järvinen 		if (uart_lsr_tx_empty(status))
111b334214eSIlpo Järvinen 			break;
112b334214eSIlpo Järvinen 		cpu_relax();
113b334214eSIlpo Järvinen 	}
114b334214eSIlpo Järvinen }
115b334214eSIlpo Järvinen 
au_early_serial8250_write(struct console * console,const char * s,unsigned int count)116b334214eSIlpo Järvinen static void au_early_serial8250_write(struct console *console,
117b334214eSIlpo Järvinen 				      const char *s, unsigned int count)
118b334214eSIlpo Järvinen {
119b334214eSIlpo Järvinen 	struct earlycon_device *device = console->data;
120b334214eSIlpo Järvinen 	struct uart_port *port = &device->port;
121b334214eSIlpo Järvinen 
122b334214eSIlpo Järvinen 	uart_console_write(port, s, count, au_putc);
123b334214eSIlpo Järvinen }
124b334214eSIlpo Järvinen 
early_au_setup(struct earlycon_device * dev,const char * opt)125b334214eSIlpo Järvinen static int __init early_au_setup(struct earlycon_device *dev, const char *opt)
126b334214eSIlpo Järvinen {
127b334214eSIlpo Järvinen 	rt288x_setup(&dev->port);
128b334214eSIlpo Järvinen 	dev->con->write = au_early_serial8250_write;
129b334214eSIlpo Järvinen 
130b334214eSIlpo Järvinen 	return 0;
131b334214eSIlpo Järvinen }
132b334214eSIlpo Järvinen OF_EARLYCON_DECLARE(palmchip, "ralink,rt2880-uart", early_au_setup);
133b334214eSIlpo Järvinen #endif
134b334214eSIlpo Järvinen 
135b334214eSIlpo Järvinen MODULE_DESCRIPTION("RT288x/Au1xxx UART driver");
136b334214eSIlpo Järvinen MODULE_LICENSE("GPL");
137