2002-04-01 14:29:03 +00:00
|
|
|
/*
|
|
|
|
* COM1 NS16550 support
|
2010-04-15 14:07:28 +00:00
|
|
|
* originally from linux source (arch/powerpc/boot/ns16550.c)
|
2008-10-16 13:01:15 +00:00
|
|
|
* modified to use CONFIG_SYS_ISA_MEM and new defines
|
2002-04-01 14:29:03 +00:00
|
|
|
*/
|
|
|
|
|
2014-09-04 22:27:32 +00:00
|
|
|
#include <common.h>
|
2014-09-04 22:27:34 +00:00
|
|
|
#include <dm.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <fdtdec.h>
|
2002-04-01 14:29:03 +00:00
|
|
|
#include <ns16550.h>
|
2014-09-04 22:27:34 +00:00
|
|
|
#include <serial.h>
|
2010-02-01 22:34:25 +00:00
|
|
|
#include <watchdog.h>
|
2010-04-23 14:05:46 +00:00
|
|
|
#include <linux/types.h>
|
|
|
|
#include <asm/io.h>
|
2002-04-01 14:29:03 +00:00
|
|
|
|
2014-09-04 22:27:34 +00:00
|
|
|
DECLARE_GLOBAL_DATA_PTR;
|
|
|
|
|
2009-04-03 09:53:01 +00:00
|
|
|
#define UART_LCRVAL UART_LCR_8N1 /* 8 data, 1 stop, no parity */
|
|
|
|
#define UART_MCRVAL (UART_MCR_DTR | \
|
|
|
|
UART_MCR_RTS) /* RTS/DTR */
|
|
|
|
#define UART_FCRVAL (UART_FCR_FIFO_EN | \
|
|
|
|
UART_FCR_RXSR | \
|
|
|
|
UART_FCR_TXSR) /* Clear & enable FIFOs */
|
2014-09-04 22:27:34 +00:00
|
|
|
|
|
|
|
#ifndef CONFIG_DM_SERIAL
|
2010-04-23 14:05:46 +00:00
|
|
|
#ifdef CONFIG_SYS_NS16550_PORT_MAPPED
|
2011-10-15 19:14:09 +00:00
|
|
|
#define serial_out(x, y) outb(x, (ulong)y)
|
|
|
|
#define serial_in(y) inb((ulong)y)
|
2011-09-01 22:47:14 +00:00
|
|
|
#elif defined(CONFIG_SYS_NS16550_MEM32) && (CONFIG_SYS_NS16550_REG_SIZE > 0)
|
2011-10-15 19:14:09 +00:00
|
|
|
#define serial_out(x, y) out_be32(y, x)
|
|
|
|
#define serial_in(y) in_be32(y)
|
2011-09-01 22:47:14 +00:00
|
|
|
#elif defined(CONFIG_SYS_NS16550_MEM32) && (CONFIG_SYS_NS16550_REG_SIZE < 0)
|
2011-10-15 19:14:09 +00:00
|
|
|
#define serial_out(x, y) out_le32(y, x)
|
|
|
|
#define serial_in(y) in_le32(y)
|
2010-04-23 14:05:46 +00:00
|
|
|
#else
|
2011-10-15 19:14:09 +00:00
|
|
|
#define serial_out(x, y) writeb(x, y)
|
|
|
|
#define serial_in(y) readb(y)
|
2010-04-23 14:05:46 +00:00
|
|
|
#endif
|
2014-09-04 22:27:34 +00:00
|
|
|
#endif /* !CONFIG_DM_SERIAL */
|
2002-04-01 14:29:03 +00:00
|
|
|
|
2014-07-15 21:59:25 +00:00
|
|
|
#if defined(CONFIG_SOC_KEYSTONE)
|
2014-04-04 17:16:53 +00:00
|
|
|
#define UART_REG_VAL_PWREMU_MGMT_UART_DISABLE 0
|
|
|
|
#define UART_REG_VAL_PWREMU_MGMT_UART_ENABLE ((1 << 14) | (1 << 13) | (1 << 0))
|
2014-04-09 19:38:46 +00:00
|
|
|
#undef UART_MCRVAL
|
|
|
|
#ifdef CONFIG_SERIAL_HW_FLOW_CONTROL
|
|
|
|
#define UART_MCRVAL (UART_MCR_RTS | UART_MCR_AFE)
|
|
|
|
#else
|
|
|
|
#define UART_MCRVAL (UART_MCR_RTS)
|
|
|
|
#endif
|
2014-04-04 17:16:53 +00:00
|
|
|
#endif
|
|
|
|
|
2010-10-27 16:28:31 +00:00
|
|
|
#ifndef CONFIG_SYS_NS16550_IER
|
|
|
|
#define CONFIG_SYS_NS16550_IER 0x00
|
|
|
|
#endif /* CONFIG_SYS_NS16550_IER */
|
|
|
|
|
2014-09-04 22:27:34 +00:00
|
|
|
#ifdef CONFIG_DM_SERIAL
|
|
|
|
|
ns16550: add generic binding to unify the drivers
Add generic binding to unify ns16550 drivers. There are
several drivers using almost the same code, such as serial_dw,
serial_keystone, serial_omap, serial_ppc, serial_rockchip,
serial_tegra.c, and serial_x86. But each is platform specific.
The key difference between these drivers is the way to get
input clock frequency. With this unified approach, fixed clock
frequency should be extracted from "clock-frequency" property of
device tree blob. If this property is not available, the macro
CONFIG_SYS_NS16550_CLK will be used. It can be a constant or a
function to get clock, eg, get_serial_clock().
Signed-off-by: Thomas Chou <thomas@wytron.com.tw>
Reviewed-by: Tom Rini <trini@konsulko.com>
Reviewed-by: Bin Meng <bmeng.cn@gmail.com>
Reviewed-by: Heiko Schocher <hs@denx.de>
Acked-by: Simon Glass <sjg@chromium.org>
2015-11-19 13:48:05 +00:00
|
|
|
#ifndef CONFIG_SYS_NS16550_CLK
|
|
|
|
#define CONFIG_SYS_NS16550_CLK 0
|
|
|
|
#endif
|
|
|
|
|
2015-02-28 05:06:26 +00:00
|
|
|
static inline void serial_out_shift(void *addr, int shift, int value)
|
2015-01-27 01:27:08 +00:00
|
|
|
{
|
2014-09-04 22:27:34 +00:00
|
|
|
#ifdef CONFIG_SYS_NS16550_PORT_MAPPED
|
2014-10-10 13:49:13 +00:00
|
|
|
outb(value, (ulong)addr);
|
2014-09-04 22:27:34 +00:00
|
|
|
#elif defined(CONFIG_SYS_NS16550_MEM32) && !defined(CONFIG_SYS_BIG_ENDIAN)
|
|
|
|
out_le32(addr, value);
|
|
|
|
#elif defined(CONFIG_SYS_NS16550_MEM32) && defined(CONFIG_SYS_BIG_ENDIAN)
|
|
|
|
out_be32(addr, value);
|
2015-05-12 20:55:02 +00:00
|
|
|
#elif defined(CONFIG_SYS_NS16550_MEM32)
|
|
|
|
writel(value, addr);
|
2014-09-04 22:27:34 +00:00
|
|
|
#elif defined(CONFIG_SYS_BIG_ENDIAN)
|
2015-01-27 01:27:08 +00:00
|
|
|
writeb(value, addr + (1 << shift) - 1);
|
2014-09-04 22:27:34 +00:00
|
|
|
#else
|
|
|
|
writeb(value, addr);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-02-28 05:06:26 +00:00
|
|
|
static inline int serial_in_shift(void *addr, int shift)
|
2014-09-04 22:27:34 +00:00
|
|
|
{
|
|
|
|
#ifdef CONFIG_SYS_NS16550_PORT_MAPPED
|
2014-10-10 13:49:13 +00:00
|
|
|
return inb((ulong)addr);
|
2014-09-04 22:27:34 +00:00
|
|
|
#elif defined(CONFIG_SYS_NS16550_MEM32) && !defined(CONFIG_SYS_BIG_ENDIAN)
|
|
|
|
return in_le32(addr);
|
|
|
|
#elif defined(CONFIG_SYS_NS16550_MEM32) && defined(CONFIG_SYS_BIG_ENDIAN)
|
|
|
|
return in_be32(addr);
|
2015-05-12 20:55:02 +00:00
|
|
|
#elif defined(CONFIG_SYS_NS16550_MEM32)
|
|
|
|
return readl(addr);
|
2014-09-04 22:27:34 +00:00
|
|
|
#elif defined(CONFIG_SYS_BIG_ENDIAN)
|
2015-02-28 07:55:36 +00:00
|
|
|
return readb(addr + (1 << shift) - 1);
|
2014-09-04 22:27:34 +00:00
|
|
|
#else
|
|
|
|
return readb(addr);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-01-27 01:27:08 +00:00
|
|
|
static void ns16550_writeb(NS16550_t port, int offset, int value)
|
|
|
|
{
|
|
|
|
struct ns16550_platdata *plat = port->plat;
|
|
|
|
unsigned char *addr;
|
|
|
|
|
|
|
|
offset *= 1 << plat->reg_shift;
|
2015-11-19 13:48:04 +00:00
|
|
|
addr = map_physmem(plat->base, 0, MAP_NOCACHE) + offset;
|
2015-01-27 01:27:08 +00:00
|
|
|
/*
|
|
|
|
* As far as we know it doesn't make sense to support selection of
|
|
|
|
* these options at run-time, so use the existing CONFIG options.
|
|
|
|
*/
|
|
|
|
serial_out_shift(addr, plat->reg_shift, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ns16550_readb(NS16550_t port, int offset)
|
|
|
|
{
|
|
|
|
struct ns16550_platdata *plat = port->plat;
|
|
|
|
unsigned char *addr;
|
|
|
|
|
|
|
|
offset *= 1 << plat->reg_shift;
|
2015-11-19 13:48:04 +00:00
|
|
|
addr = map_physmem(plat->base, 0, MAP_NOCACHE) + offset;
|
2015-01-27 01:27:08 +00:00
|
|
|
|
|
|
|
return serial_in_shift(addr, plat->reg_shift);
|
|
|
|
}
|
|
|
|
|
2014-09-04 22:27:34 +00:00
|
|
|
/* We can clean these up once everything is moved to driver model */
|
|
|
|
#define serial_out(value, addr) \
|
2015-02-28 05:06:26 +00:00
|
|
|
ns16550_writeb(com_port, \
|
|
|
|
(unsigned char *)addr - (unsigned char *)com_port, value)
|
2014-09-04 22:27:34 +00:00
|
|
|
#define serial_in(addr) \
|
2015-02-28 05:06:26 +00:00
|
|
|
ns16550_readb(com_port, \
|
|
|
|
(unsigned char *)addr - (unsigned char *)com_port)
|
2014-09-04 22:27:34 +00:00
|
|
|
#endif
|
|
|
|
|
2015-01-27 01:27:09 +00:00
|
|
|
static inline int calc_divisor(NS16550_t port, int clock, int baudrate)
|
2014-09-04 22:27:32 +00:00
|
|
|
{
|
|
|
|
const unsigned int mode_x_div = 16;
|
|
|
|
|
2015-01-27 01:27:09 +00:00
|
|
|
return DIV_ROUND_CLOSEST(clock, mode_x_div * baudrate);
|
|
|
|
}
|
|
|
|
|
|
|
|
int ns16550_calc_divisor(NS16550_t port, int clock, int baudrate)
|
|
|
|
{
|
2014-09-04 22:27:32 +00:00
|
|
|
#ifdef CONFIG_OMAP1510
|
|
|
|
/* If can't cleanly clock 115200 set div to 1 */
|
|
|
|
if ((clock == 12000000) && (baudrate == 115200)) {
|
|
|
|
port->osc_12m_sel = OSC_12M_SEL; /* enable 6.5 * divisor */
|
|
|
|
return 1; /* return 1 for base divisor */
|
|
|
|
}
|
|
|
|
port->osc_12m_sel = 0; /* clear if previsouly set */
|
|
|
|
#endif
|
|
|
|
|
2015-01-27 01:27:09 +00:00
|
|
|
return calc_divisor(port, clock, baudrate);
|
2014-09-04 22:27:32 +00:00
|
|
|
}
|
|
|
|
|
2014-09-04 22:27:33 +00:00
|
|
|
static void NS16550_setbrg(NS16550_t com_port, int baud_divisor)
|
|
|
|
{
|
|
|
|
serial_out(UART_LCR_BKSE | UART_LCRVAL, &com_port->lcr);
|
|
|
|
serial_out(baud_divisor & 0xff, &com_port->dll);
|
|
|
|
serial_out((baud_divisor >> 8) & 0xff, &com_port->dlm);
|
|
|
|
serial_out(UART_LCRVAL, &com_port->lcr);
|
|
|
|
}
|
|
|
|
|
2011-10-15 19:14:09 +00:00
|
|
|
void NS16550_init(NS16550_t com_port, int baud_divisor)
|
2002-04-01 14:29:03 +00:00
|
|
|
{
|
2014-11-10 19:04:10 +00:00
|
|
|
#if (defined(CONFIG_SPL_BUILD) && \
|
|
|
|
(defined(CONFIG_OMAP34XX) || defined(CONFIG_OMAP44XX)))
|
2013-03-29 02:52:36 +00:00
|
|
|
/*
|
2014-11-10 19:04:10 +00:00
|
|
|
* On some OMAP3/OMAP4 devices when UART3 is configured for boot mode
|
|
|
|
* before SPL starts only THRE bit is set. We have to empty the
|
|
|
|
* transmitter before initialization starts.
|
2013-03-29 02:52:36 +00:00
|
|
|
*/
|
|
|
|
if ((serial_in(&com_port->lsr) & (UART_LSR_TEMT | UART_LSR_THRE))
|
|
|
|
== UART_LSR_THRE) {
|
2014-09-04 22:27:34 +00:00
|
|
|
if (baud_divisor != -1)
|
|
|
|
NS16550_setbrg(com_port, baud_divisor);
|
2013-03-29 02:52:36 +00:00
|
|
|
serial_out(0, &com_port->mdr1);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2012-09-18 23:19:05 +00:00
|
|
|
while (!(serial_in(&com_port->lsr) & UART_LSR_TEMT))
|
|
|
|
;
|
|
|
|
|
2010-10-27 16:28:31 +00:00
|
|
|
serial_out(CONFIG_SYS_NS16550_IER, &com_port->ier);
|
2013-12-20 16:19:33 +00:00
|
|
|
#if defined(CONFIG_OMAP) || defined(CONFIG_AM33XX) || \
|
|
|
|
defined(CONFIG_TI81XX) || defined(CONFIG_AM43XX)
|
2010-04-23 14:05:46 +00:00
|
|
|
serial_out(0x7, &com_port->mdr1); /* mode select reset TL16C750*/
|
2003-07-16 21:53:01 +00:00
|
|
|
#endif
|
2010-04-23 14:05:46 +00:00
|
|
|
serial_out(UART_MCRVAL, &com_port->mcr);
|
|
|
|
serial_out(UART_FCRVAL, &com_port->fcr);
|
2014-09-04 22:27:34 +00:00
|
|
|
if (baud_divisor != -1)
|
|
|
|
NS16550_setbrg(com_port, baud_divisor);
|
2014-07-30 10:11:41 +00:00
|
|
|
#if defined(CONFIG_OMAP) || \
|
2013-03-15 10:07:09 +00:00
|
|
|
defined(CONFIG_AM33XX) || defined(CONFIG_SOC_DA8XX) || \
|
2013-07-02 10:05:58 +00:00
|
|
|
defined(CONFIG_TI81XX) || defined(CONFIG_AM43XX)
|
2011-10-14 02:58:26 +00:00
|
|
|
|
2011-10-15 19:14:09 +00:00
|
|
|
/* /16 is proper to hit 115200 with 48MHz */
|
|
|
|
serial_out(0, &com_port->mdr1);
|
2009-02-12 01:26:52 +00:00
|
|
|
#endif /* CONFIG_OMAP */
|
2014-07-15 21:59:25 +00:00
|
|
|
#if defined(CONFIG_SOC_KEYSTONE)
|
2014-04-04 17:16:53 +00:00
|
|
|
serial_out(UART_REG_VAL_PWREMU_MGMT_UART_ENABLE, &com_port->regC);
|
|
|
|
#endif
|
2002-04-01 14:29:03 +00:00
|
|
|
}
|
|
|
|
|
2009-02-18 22:30:44 +00:00
|
|
|
#ifndef CONFIG_NS16550_MIN_FUNCTIONS
|
2011-10-15 19:14:09 +00:00
|
|
|
void NS16550_reinit(NS16550_t com_port, int baud_divisor)
|
2002-04-01 14:29:03 +00:00
|
|
|
{
|
2010-10-27 16:28:31 +00:00
|
|
|
serial_out(CONFIG_SYS_NS16550_IER, &com_port->ier);
|
2014-09-04 22:27:33 +00:00
|
|
|
NS16550_setbrg(com_port, 0);
|
2010-04-23 14:05:46 +00:00
|
|
|
serial_out(UART_MCRVAL, &com_port->mcr);
|
|
|
|
serial_out(UART_FCRVAL, &com_port->fcr);
|
2014-09-04 22:27:33 +00:00
|
|
|
NS16550_setbrg(com_port, baud_divisor);
|
2002-04-01 14:29:03 +00:00
|
|
|
}
|
2009-02-18 22:30:44 +00:00
|
|
|
#endif /* CONFIG_NS16550_MIN_FUNCTIONS */
|
2002-04-01 14:29:03 +00:00
|
|
|
|
2011-10-15 19:14:09 +00:00
|
|
|
void NS16550_putc(NS16550_t com_port, char c)
|
2002-04-01 14:29:03 +00:00
|
|
|
{
|
2011-10-15 19:14:09 +00:00
|
|
|
while ((serial_in(&com_port->lsr) & UART_LSR_THRE) == 0)
|
|
|
|
;
|
2010-04-23 14:05:46 +00:00
|
|
|
serial_out(c, &com_port->thr);
|
2010-10-12 07:39:45 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Call watchdog_reset() upon newline. This is done here in putc
|
|
|
|
* since the environment code uses a single puts() to print the complete
|
|
|
|
* environment upon "printenv". So we can't put this watchdog call
|
|
|
|
* in puts().
|
|
|
|
*/
|
|
|
|
if (c == '\n')
|
|
|
|
WATCHDOG_RESET();
|
2002-04-01 14:29:03 +00:00
|
|
|
}
|
|
|
|
|
2009-02-18 22:30:44 +00:00
|
|
|
#ifndef CONFIG_NS16550_MIN_FUNCTIONS
|
2011-10-15 19:14:09 +00:00
|
|
|
char NS16550_getc(NS16550_t com_port)
|
2002-04-01 14:29:03 +00:00
|
|
|
{
|
2010-04-23 14:05:46 +00:00
|
|
|
while ((serial_in(&com_port->lsr) & UART_LSR_DR) == 0) {
|
2012-09-15 08:25:19 +00:00
|
|
|
#if !defined(CONFIG_SPL_BUILD) && defined(CONFIG_USB_TTY)
|
2004-03-12 00:14:09 +00:00
|
|
|
extern void usbtty_poll(void);
|
|
|
|
usbtty_poll();
|
|
|
|
#endif
|
2010-02-01 22:34:25 +00:00
|
|
|
WATCHDOG_RESET();
|
2004-03-12 00:14:09 +00:00
|
|
|
}
|
2010-04-23 14:05:46 +00:00
|
|
|
return serial_in(&com_port->rbr);
|
2002-04-01 14:29:03 +00:00
|
|
|
}
|
|
|
|
|
2011-10-15 19:14:09 +00:00
|
|
|
int NS16550_tstc(NS16550_t com_port)
|
2002-04-01 14:29:03 +00:00
|
|
|
{
|
2011-10-15 19:14:09 +00:00
|
|
|
return (serial_in(&com_port->lsr) & UART_LSR_DR) != 0;
|
2002-04-01 14:29:03 +00:00
|
|
|
}
|
|
|
|
|
2009-02-18 22:30:44 +00:00
|
|
|
#endif /* CONFIG_NS16550_MIN_FUNCTIONS */
|
2014-09-04 22:27:34 +00:00
|
|
|
|
2015-01-27 01:27:09 +00:00
|
|
|
#ifdef CONFIG_DEBUG_UART_NS16550
|
|
|
|
|
|
|
|
#include <debug_uart.h>
|
|
|
|
|
2015-06-23 21:39:06 +00:00
|
|
|
#define serial_dout(reg, value) \
|
|
|
|
serial_out_shift((char *)com_port + \
|
|
|
|
((char *)reg - (char *)com_port) * \
|
|
|
|
(1 << CONFIG_DEBUG_UART_SHIFT), \
|
|
|
|
CONFIG_DEBUG_UART_SHIFT, value)
|
|
|
|
#define serial_din(reg) \
|
|
|
|
serial_in_shift((char *)com_port + \
|
|
|
|
((char *)reg - (char *)com_port) * \
|
|
|
|
(1 << CONFIG_DEBUG_UART_SHIFT), \
|
|
|
|
CONFIG_DEBUG_UART_SHIFT)
|
|
|
|
|
2015-10-19 01:51:23 +00:00
|
|
|
static inline void _debug_uart_init(void)
|
2015-01-27 01:27:09 +00:00
|
|
|
{
|
|
|
|
struct NS16550 *com_port = (struct NS16550 *)CONFIG_DEBUG_UART_BASE;
|
|
|
|
int baud_divisor;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We copy the code from above because it is already horribly messy.
|
|
|
|
* Trying to refactor to nicely remove the duplication doesn't seem
|
|
|
|
* feasible. The better fix is to move all users of this driver to
|
|
|
|
* driver model.
|
|
|
|
*/
|
|
|
|
baud_divisor = calc_divisor(com_port, CONFIG_DEBUG_UART_CLOCK,
|
|
|
|
CONFIG_BAUDRATE);
|
2015-06-23 21:39:06 +00:00
|
|
|
serial_dout(&com_port->ier, CONFIG_SYS_NS16550_IER);
|
|
|
|
serial_dout(&com_port->mcr, UART_MCRVAL);
|
|
|
|
serial_dout(&com_port->fcr, UART_FCRVAL);
|
|
|
|
|
|
|
|
serial_dout(&com_port->lcr, UART_LCR_BKSE | UART_LCRVAL);
|
|
|
|
serial_dout(&com_port->dll, baud_divisor & 0xff);
|
|
|
|
serial_dout(&com_port->dlm, (baud_divisor >> 8) & 0xff);
|
|
|
|
serial_dout(&com_port->lcr, UART_LCRVAL);
|
2015-01-27 01:27:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void _debug_uart_putc(int ch)
|
|
|
|
{
|
|
|
|
struct NS16550 *com_port = (struct NS16550 *)CONFIG_DEBUG_UART_BASE;
|
|
|
|
|
2015-06-23 21:39:06 +00:00
|
|
|
while (!(serial_din(&com_port->lsr) & UART_LSR_THRE))
|
2015-01-27 01:27:09 +00:00
|
|
|
;
|
2015-06-23 21:39:06 +00:00
|
|
|
serial_dout(&com_port->thr, ch);
|
2015-01-27 01:27:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG_UART_FUNCS
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2014-09-04 22:27:34 +00:00
|
|
|
#ifdef CONFIG_DM_SERIAL
|
|
|
|
static int ns16550_serial_putc(struct udevice *dev, const char ch)
|
|
|
|
{
|
|
|
|
struct NS16550 *const com_port = dev_get_priv(dev);
|
|
|
|
|
|
|
|
if (!(serial_in(&com_port->lsr) & UART_LSR_THRE))
|
|
|
|
return -EAGAIN;
|
|
|
|
serial_out(ch, &com_port->thr);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Call watchdog_reset() upon newline. This is done here in putc
|
|
|
|
* since the environment code uses a single puts() to print the complete
|
|
|
|
* environment upon "printenv". So we can't put this watchdog call
|
|
|
|
* in puts().
|
|
|
|
*/
|
|
|
|
if (ch == '\n')
|
|
|
|
WATCHDOG_RESET();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ns16550_serial_pending(struct udevice *dev, bool input)
|
|
|
|
{
|
|
|
|
struct NS16550 *const com_port = dev_get_priv(dev);
|
|
|
|
|
|
|
|
if (input)
|
|
|
|
return serial_in(&com_port->lsr) & UART_LSR_DR ? 1 : 0;
|
|
|
|
else
|
|
|
|
return serial_in(&com_port->lsr) & UART_LSR_THRE ? 0 : 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ns16550_serial_getc(struct udevice *dev)
|
|
|
|
{
|
|
|
|
struct NS16550 *const com_port = dev_get_priv(dev);
|
|
|
|
|
2014-10-23 03:37:03 +00:00
|
|
|
if (!(serial_in(&com_port->lsr) & UART_LSR_DR))
|
2014-09-04 22:27:34 +00:00
|
|
|
return -EAGAIN;
|
|
|
|
|
|
|
|
return serial_in(&com_port->rbr);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ns16550_serial_setbrg(struct udevice *dev, int baudrate)
|
|
|
|
{
|
|
|
|
struct NS16550 *const com_port = dev_get_priv(dev);
|
|
|
|
struct ns16550_platdata *plat = com_port->plat;
|
|
|
|
int clock_divisor;
|
|
|
|
|
|
|
|
clock_divisor = ns16550_calc_divisor(com_port, plat->clock, baudrate);
|
|
|
|
|
|
|
|
NS16550_setbrg(com_port, clock_divisor);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ns16550_serial_probe(struct udevice *dev)
|
|
|
|
{
|
|
|
|
struct NS16550 *const com_port = dev_get_priv(dev);
|
|
|
|
|
2014-10-23 03:37:05 +00:00
|
|
|
com_port->plat = dev_get_platdata(dev);
|
2014-09-04 22:27:34 +00:00
|
|
|
NS16550_init(com_port, -1);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-08-11 22:31:55 +00:00
|
|
|
#if CONFIG_IS_ENABLED(OF_CONTROL)
|
2014-09-04 22:27:34 +00:00
|
|
|
int ns16550_serial_ofdata_to_platdata(struct udevice *dev)
|
|
|
|
{
|
|
|
|
struct ns16550_platdata *plat = dev->platdata;
|
|
|
|
fdt_addr_t addr;
|
|
|
|
|
2014-12-31 08:05:12 +00:00
|
|
|
/* try Processor Local Bus device first */
|
2015-08-11 14:33:29 +00:00
|
|
|
addr = dev_get_addr(dev);
|
2015-11-29 20:17:54 +00:00
|
|
|
#if defined(CONFIG_PCI) && defined(CONFIG_DM_PCI)
|
2014-12-31 08:05:12 +00:00
|
|
|
if (addr == FDT_ADDR_T_NONE) {
|
|
|
|
/* then try pci device */
|
|
|
|
struct fdt_pci_addr pci_addr;
|
|
|
|
u32 bar;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* we prefer to use a memory-mapped register */
|
|
|
|
ret = fdtdec_get_pci_addr(gd->fdt_blob, dev->of_offset,
|
|
|
|
FDT_PCI_SPACE_MEM32, "reg",
|
|
|
|
&pci_addr);
|
|
|
|
if (ret) {
|
|
|
|
/* try if there is any i/o-mapped register */
|
|
|
|
ret = fdtdec_get_pci_addr(gd->fdt_blob,
|
|
|
|
dev->of_offset,
|
|
|
|
FDT_PCI_SPACE_IO,
|
|
|
|
"reg", &pci_addr);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-11-29 20:17:54 +00:00
|
|
|
ret = fdtdec_get_pci_bar32(dev, &pci_addr, &bar);
|
2014-12-31 08:05:12 +00:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
addr = bar;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2014-09-04 22:27:34 +00:00
|
|
|
if (addr == FDT_ADDR_T_NONE)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2014-10-23 03:37:04 +00:00
|
|
|
plat->base = addr;
|
2014-09-04 22:27:34 +00:00
|
|
|
plat->reg_shift = fdtdec_get_int(gd->fdt_blob, dev->of_offset,
|
2015-11-29 06:01:03 +00:00
|
|
|
"reg-shift", 0);
|
ns16550: add generic binding to unify the drivers
Add generic binding to unify ns16550 drivers. There are
several drivers using almost the same code, such as serial_dw,
serial_keystone, serial_omap, serial_ppc, serial_rockchip,
serial_tegra.c, and serial_x86. But each is platform specific.
The key difference between these drivers is the way to get
input clock frequency. With this unified approach, fixed clock
frequency should be extracted from "clock-frequency" property of
device tree blob. If this property is not available, the macro
CONFIG_SYS_NS16550_CLK will be used. It can be a constant or a
function to get clock, eg, get_serial_clock().
Signed-off-by: Thomas Chou <thomas@wytron.com.tw>
Reviewed-by: Tom Rini <trini@konsulko.com>
Reviewed-by: Bin Meng <bmeng.cn@gmail.com>
Reviewed-by: Heiko Schocher <hs@denx.de>
Acked-by: Simon Glass <sjg@chromium.org>
2015-11-19 13:48:05 +00:00
|
|
|
plat->clock = fdtdec_get_int(gd->fdt_blob, dev->of_offset,
|
|
|
|
"clock-frequency",
|
|
|
|
CONFIG_SYS_NS16550_CLK);
|
|
|
|
if (!plat->clock) {
|
|
|
|
debug("ns16550 clock not defined\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2014-09-04 22:27:34 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2014-10-23 03:37:05 +00:00
|
|
|
#endif
|
2014-09-04 22:27:34 +00:00
|
|
|
|
|
|
|
const struct dm_serial_ops ns16550_serial_ops = {
|
|
|
|
.putc = ns16550_serial_putc,
|
|
|
|
.pending = ns16550_serial_pending,
|
|
|
|
.getc = ns16550_serial_getc,
|
|
|
|
.setbrg = ns16550_serial_setbrg,
|
|
|
|
};
|
ns16550: add generic binding to unify the drivers
Add generic binding to unify ns16550 drivers. There are
several drivers using almost the same code, such as serial_dw,
serial_keystone, serial_omap, serial_ppc, serial_rockchip,
serial_tegra.c, and serial_x86. But each is platform specific.
The key difference between these drivers is the way to get
input clock frequency. With this unified approach, fixed clock
frequency should be extracted from "clock-frequency" property of
device tree blob. If this property is not available, the macro
CONFIG_SYS_NS16550_CLK will be used. It can be a constant or a
function to get clock, eg, get_serial_clock().
Signed-off-by: Thomas Chou <thomas@wytron.com.tw>
Reviewed-by: Tom Rini <trini@konsulko.com>
Reviewed-by: Bin Meng <bmeng.cn@gmail.com>
Reviewed-by: Heiko Schocher <hs@denx.de>
Acked-by: Simon Glass <sjg@chromium.org>
2015-11-19 13:48:05 +00:00
|
|
|
|
|
|
|
#if CONFIG_IS_ENABLED(OF_CONTROL)
|
|
|
|
static const struct udevice_id ns16550_serial_ids[] = {
|
|
|
|
{ .compatible = "ns16550" },
|
|
|
|
{ .compatible = "ns16550a" },
|
|
|
|
{ .compatible = "nvidia,tegra20-uart" },
|
2015-11-17 06:20:14 +00:00
|
|
|
{ .compatible = "rockchip,rk3036-uart" },
|
ns16550: add generic binding to unify the drivers
Add generic binding to unify ns16550 drivers. There are
several drivers using almost the same code, such as serial_dw,
serial_keystone, serial_omap, serial_ppc, serial_rockchip,
serial_tegra.c, and serial_x86. But each is platform specific.
The key difference between these drivers is the way to get
input clock frequency. With this unified approach, fixed clock
frequency should be extracted from "clock-frequency" property of
device tree blob. If this property is not available, the macro
CONFIG_SYS_NS16550_CLK will be used. It can be a constant or a
function to get clock, eg, get_serial_clock().
Signed-off-by: Thomas Chou <thomas@wytron.com.tw>
Reviewed-by: Tom Rini <trini@konsulko.com>
Reviewed-by: Bin Meng <bmeng.cn@gmail.com>
Reviewed-by: Heiko Schocher <hs@denx.de>
Acked-by: Simon Glass <sjg@chromium.org>
2015-11-19 13:48:05 +00:00
|
|
|
{ .compatible = "snps,dw-apb-uart" },
|
|
|
|
{ .compatible = "ti,omap2-uart" },
|
|
|
|
{ .compatible = "ti,omap3-uart" },
|
|
|
|
{ .compatible = "ti,omap4-uart" },
|
|
|
|
{ .compatible = "ti,am3352-uart" },
|
|
|
|
{ .compatible = "ti,am4372-uart" },
|
|
|
|
{ .compatible = "ti,dra742-uart" },
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
U_BOOT_DRIVER(ns16550_serial) = {
|
|
|
|
.name = "ns16550_serial",
|
|
|
|
.id = UCLASS_SERIAL,
|
|
|
|
#if CONFIG_IS_ENABLED(OF_CONTROL)
|
|
|
|
.of_match = ns16550_serial_ids,
|
|
|
|
.ofdata_to_platdata = ns16550_serial_ofdata_to_platdata,
|
|
|
|
.platdata_auto_alloc_size = sizeof(struct ns16550_platdata),
|
|
|
|
#endif
|
|
|
|
.priv_auto_alloc_size = sizeof(struct NS16550),
|
|
|
|
.probe = ns16550_serial_probe,
|
|
|
|
.ops = &ns16550_serial_ops,
|
2015-12-04 15:58:38 +00:00
|
|
|
.flags = DM_FLAG_PRE_RELOC,
|
ns16550: add generic binding to unify the drivers
Add generic binding to unify ns16550 drivers. There are
several drivers using almost the same code, such as serial_dw,
serial_keystone, serial_omap, serial_ppc, serial_rockchip,
serial_tegra.c, and serial_x86. But each is platform specific.
The key difference between these drivers is the way to get
input clock frequency. With this unified approach, fixed clock
frequency should be extracted from "clock-frequency" property of
device tree blob. If this property is not available, the macro
CONFIG_SYS_NS16550_CLK will be used. It can be a constant or a
function to get clock, eg, get_serial_clock().
Signed-off-by: Thomas Chou <thomas@wytron.com.tw>
Reviewed-by: Tom Rini <trini@konsulko.com>
Reviewed-by: Bin Meng <bmeng.cn@gmail.com>
Reviewed-by: Heiko Schocher <hs@denx.de>
Acked-by: Simon Glass <sjg@chromium.org>
2015-11-19 13:48:05 +00:00
|
|
|
};
|
2014-09-04 22:27:34 +00:00
|
|
|
#endif /* CONFIG_DM_SERIAL */
|