/*
|
* (C) Copyright 2004, Psyent Corporation <www.psyent.com>
|
* Scott McNutt <smcnutt@psyent.com>
|
*
|
* SPDX-License-Identifier: GPL-2.0+
|
*/
|
|
#include <common.h>
|
#include <dm.h>
|
#include <errno.h>
|
#include <serial.h>
|
#include <asm/io.h>
|
|
DECLARE_GLOBAL_DATA_PTR;
|
|
/* data register */
|
#define ALTERA_JTAG_RVALID BIT(15) /* Read valid */
|
|
/* control register */
|
#define ALTERA_JTAG_AC BIT(10) /* activity indicator */
|
#define ALTERA_JTAG_RRDY BIT(12) /* read available */
|
#define ALTERA_JTAG_WSPACE(d) ((d) >> 16) /* Write space avail */
|
/* Write fifo size. FIXME: this should be extracted with sopc2dts */
|
#define ALTERA_JTAG_WRITE_DEPTH 64
|
|
struct altera_jtaguart_regs {
|
u32 data; /* Data register */
|
u32 control; /* Control register */
|
};
|
|
struct altera_jtaguart_platdata {
|
struct altera_jtaguart_regs *regs;
|
};
|
|
static int altera_jtaguart_setbrg(struct udevice *dev, int baudrate)
|
{
|
return 0;
|
}
|
|
static int altera_jtaguart_putc(struct udevice *dev, const char ch)
|
{
|
struct altera_jtaguart_platdata *plat = dev->platdata;
|
struct altera_jtaguart_regs *const regs = plat->regs;
|
u32 st = readl(®s->control);
|
|
#ifdef CONFIG_ALTERA_JTAG_UART_BYPASS
|
if (!(st & ALTERA_JTAG_AC)) /* no connection yet */
|
return -ENETUNREACH;
|
#endif
|
|
if (ALTERA_JTAG_WSPACE(st) == 0)
|
return -EAGAIN;
|
|
writel(ch, ®s->data);
|
|
return 0;
|
}
|
|
static int altera_jtaguart_pending(struct udevice *dev, bool input)
|
{
|
struct altera_jtaguart_platdata *plat = dev->platdata;
|
struct altera_jtaguart_regs *const regs = plat->regs;
|
u32 st = readl(®s->control);
|
|
if (input)
|
return st & ALTERA_JTAG_RRDY ? 1 : 0;
|
else
|
return !(ALTERA_JTAG_WSPACE(st) == ALTERA_JTAG_WRITE_DEPTH);
|
}
|
|
static int altera_jtaguart_getc(struct udevice *dev)
|
{
|
struct altera_jtaguart_platdata *plat = dev->platdata;
|
struct altera_jtaguart_regs *const regs = plat->regs;
|
u32 val;
|
|
val = readl(®s->data);
|
|
if (!(val & ALTERA_JTAG_RVALID))
|
return -EAGAIN;
|
|
return val & 0xff;
|
}
|
|
static int altera_jtaguart_probe(struct udevice *dev)
|
{
|
#ifdef CONFIG_ALTERA_JTAG_UART_BYPASS
|
struct altera_jtaguart_platdata *plat = dev->platdata;
|
struct altera_jtaguart_regs *const regs = plat->regs;
|
|
writel(ALTERA_JTAG_AC, ®s->control); /* clear AC flag */
|
#endif
|
return 0;
|
}
|
|
static int altera_jtaguart_ofdata_to_platdata(struct udevice *dev)
|
{
|
struct altera_jtaguart_platdata *plat = dev_get_platdata(dev);
|
|
plat->regs = map_physmem(devfdt_get_addr(dev),
|
sizeof(struct altera_jtaguart_regs),
|
MAP_NOCACHE);
|
|
return 0;
|
}
|
|
static const struct dm_serial_ops altera_jtaguart_ops = {
|
.putc = altera_jtaguart_putc,
|
.pending = altera_jtaguart_pending,
|
.getc = altera_jtaguart_getc,
|
.setbrg = altera_jtaguart_setbrg,
|
};
|
|
static const struct udevice_id altera_jtaguart_ids[] = {
|
{ .compatible = "altr,juart-1.0" },
|
{}
|
};
|
|
U_BOOT_DRIVER(altera_jtaguart) = {
|
.name = "altera_jtaguart",
|
.id = UCLASS_SERIAL,
|
.of_match = altera_jtaguart_ids,
|
.ofdata_to_platdata = altera_jtaguart_ofdata_to_platdata,
|
.platdata_auto_alloc_size = sizeof(struct altera_jtaguart_platdata),
|
.probe = altera_jtaguart_probe,
|
.ops = &altera_jtaguart_ops,
|
.flags = DM_FLAG_PRE_RELOC,
|
};
|
|
#ifdef CONFIG_DEBUG_UART_ALTERA_JTAGUART
|
|
#include <debug_uart.h>
|
|
static inline void _debug_uart_init(void)
|
{
|
}
|
|
static inline void _debug_uart_putc(int ch)
|
{
|
struct altera_jtaguart_regs *regs = (void *)CONFIG_DEBUG_UART_BASE;
|
|
while (1) {
|
u32 st = readl(®s->control);
|
|
if (ALTERA_JTAG_WSPACE(st))
|
break;
|
}
|
|
writel(ch, ®s->data);
|
}
|
|
DEBUG_UART_FUNCS
|
|
#endif
|