b18300e61606bac65f9ca47897277eee8706d41e
[platform/kernel/u-boot.git] / drivers / serial / altera_jtag_uart.c
1 /*
2  * (C) Copyright 2004, Psyent Corporation <www.psyent.com>
3  * Scott McNutt <smcnutt@psyent.com>
4  *
5  * SPDX-License-Identifier:     GPL-2.0+
6  */
7
8 #include <common.h>
9 #include <dm.h>
10 #include <errno.h>
11 #include <serial.h>
12 #include <asm/io.h>
13
14 /* data register */
15 #define ALTERA_JTAG_RVALID      BIT(15) /* Read valid */
16
17 /* control register */
18 #define ALTERA_JTAG_AC          BIT(10) /* activity indicator */
19 #define ALTERA_JTAG_RRDY        BIT(12) /* read available */
20 #define ALTERA_JTAG_WSPACE(d)   ((d) >> 16)     /* Write space avail */
21 /* Write fifo size. FIXME: this should be extracted with sopc2dts */
22 #define ALTERA_JTAG_WRITE_DEPTH 64
23
24 struct altera_jtaguart_regs {
25         u32     data;                   /* Data register */
26         u32     control;                /* Control register */
27 };
28
29 struct altera_jtaguart_platdata {
30         struct altera_jtaguart_regs *regs;
31 };
32
33 static int altera_jtaguart_setbrg(struct udevice *dev, int baudrate)
34 {
35         return 0;
36 }
37
38 static int altera_jtaguart_putc(struct udevice *dev, const char ch)
39 {
40         struct altera_jtaguart_platdata *plat = dev->platdata;
41         struct altera_jtaguart_regs *const regs = plat->regs;
42         u32 st = readl(&regs->control);
43
44 #ifdef CONFIG_ALTERA_JTAG_UART_BYPASS
45         if (!(st & ALTERA_JTAG_AC)) /* no connection yet */
46                 return -ENETUNREACH;
47 #endif
48
49         if (ALTERA_JTAG_WSPACE(st) == 0)
50                 return -EAGAIN;
51
52         writel(ch, &regs->data);
53
54         return 0;
55 }
56
57 static int altera_jtaguart_pending(struct udevice *dev, bool input)
58 {
59         struct altera_jtaguart_platdata *plat = dev->platdata;
60         struct altera_jtaguart_regs *const regs = plat->regs;
61         u32 st = readl(&regs->control);
62
63         if (input)
64                 return st & ALTERA_JTAG_RRDY ? 1 : 0;
65         else
66                 return !(ALTERA_JTAG_WSPACE(st) == ALTERA_JTAG_WRITE_DEPTH);
67 }
68
69 static int altera_jtaguart_getc(struct udevice *dev)
70 {
71         struct altera_jtaguart_platdata *plat = dev->platdata;
72         struct altera_jtaguart_regs *const regs = plat->regs;
73         u32 val;
74
75         val = readl(&regs->data);
76
77         if (!(val & ALTERA_JTAG_RVALID))
78                 return -EAGAIN;
79
80         return val & 0xff;
81 }
82
83 static int altera_jtaguart_probe(struct udevice *dev)
84 {
85 #ifdef CONFIG_ALTERA_JTAG_UART_BYPASS
86         struct altera_jtaguart_platdata *plat = dev->platdata;
87         struct altera_jtaguart_regs *const regs = plat->regs;
88
89         writel(ALTERA_JTAG_AC, &regs->control); /* clear AC flag */
90 #endif
91         return 0;
92 }
93
94 static int altera_jtaguart_ofdata_to_platdata(struct udevice *dev)
95 {
96         struct altera_jtaguart_platdata *plat = dev_get_platdata(dev);
97
98         plat->regs = map_physmem(devfdt_get_addr(dev),
99                                  sizeof(struct altera_jtaguart_regs),
100                                  MAP_NOCACHE);
101
102         return 0;
103 }
104
105 static const struct dm_serial_ops altera_jtaguart_ops = {
106         .putc = altera_jtaguart_putc,
107         .pending = altera_jtaguart_pending,
108         .getc = altera_jtaguart_getc,
109         .setbrg = altera_jtaguart_setbrg,
110 };
111
112 static const struct udevice_id altera_jtaguart_ids[] = {
113         { .compatible = "altr,juart-1.0" },
114         {}
115 };
116
117 U_BOOT_DRIVER(altera_jtaguart) = {
118         .name   = "altera_jtaguart",
119         .id     = UCLASS_SERIAL,
120         .of_match = altera_jtaguart_ids,
121         .ofdata_to_platdata = altera_jtaguart_ofdata_to_platdata,
122         .platdata_auto_alloc_size = sizeof(struct altera_jtaguart_platdata),
123         .probe = altera_jtaguart_probe,
124         .ops    = &altera_jtaguart_ops,
125         .flags = DM_FLAG_PRE_RELOC,
126 };
127
128 #ifdef CONFIG_DEBUG_UART_ALTERA_JTAGUART
129
130 #include <debug_uart.h>
131
132 static inline void _debug_uart_init(void)
133 {
134 }
135
136 static inline void _debug_uart_putc(int ch)
137 {
138         struct altera_jtaguart_regs *regs = (void *)CONFIG_DEBUG_UART_BASE;
139
140         while (1) {
141                 u32 st = readl(&regs->control);
142
143                 if (ALTERA_JTAG_WSPACE(st))
144                         break;
145         }
146
147         writel(ch, &regs->data);
148 }
149
150 DEBUG_UART_FUNCS
151
152 #endif