blob: 9851663dc54f8a301fb8bb075898473621cc8538 [file] [log] [blame]
wdenke85390d2002-04-01 14:29:03 +00001/*
2 * COM1 NS16550 support
Stefan Roese88fbf932010-04-15 16:07:28 +02003 * originally from linux source (arch/powerpc/boot/ns16550.c)
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +02004 * modified to use CONFIG_SYS_ISA_MEM and new defines
wdenke85390d2002-04-01 14:29:03 +00005 */
6
Simon Glass85d65312019-12-28 10:44:58 -07007#include <clock_legacy.h>
Simon Glasse98e01e2014-09-04 16:27:32 -06008#include <common.h>
Paul Burton8aadc562016-09-08 07:47:29 +01009#include <clk.h>
Simon Glass79a9da32014-09-04 16:27:34 -060010#include <dm.h>
11#include <errno.h>
wdenke85390d2002-04-01 14:29:03 +000012#include <ns16550.h>
Ley Foon Tana1763392018-06-14 18:45:22 +080013#include <reset.h>
Simon Glass79a9da32014-09-04 16:27:34 -060014#include <serial.h>
Ladislav Michlcc294422010-02-01 23:34:25 +010015#include <watchdog.h>
Graeme Russ14f06e62010-04-24 00:05:46 +100016#include <linux/types.h>
17#include <asm/io.h>
wdenke85390d2002-04-01 14:29:03 +000018
Simon Glass79a9da32014-09-04 16:27:34 -060019DECLARE_GLOBAL_DATA_PTR;
20
Detlev Zundel166fb542009-04-03 11:53:01 +020021#define UART_LCRVAL UART_LCR_8N1 /* 8 data, 1 stop, no parity */
22#define UART_MCRVAL (UART_MCR_DTR | \
23 UART_MCR_RTS) /* RTS/DTR */
Simon Glass79a9da32014-09-04 16:27:34 -060024
Simon Glassd945a492019-09-25 08:11:14 -060025#if !CONFIG_IS_ENABLED(DM_SERIAL)
Graeme Russ14f06e62010-04-24 00:05:46 +100026#ifdef CONFIG_SYS_NS16550_PORT_MAPPED
Simon Glassdd5497c2011-10-15 19:14:09 +000027#define serial_out(x, y) outb(x, (ulong)y)
28#define serial_in(y) inb((ulong)y)
Dave Aldridgea51bebc2011-09-01 22:47:14 +000029#elif defined(CONFIG_SYS_NS16550_MEM32) && (CONFIG_SYS_NS16550_REG_SIZE > 0)
Simon Glassdd5497c2011-10-15 19:14:09 +000030#define serial_out(x, y) out_be32(y, x)
31#define serial_in(y) in_be32(y)
Dave Aldridgea51bebc2011-09-01 22:47:14 +000032#elif defined(CONFIG_SYS_NS16550_MEM32) && (CONFIG_SYS_NS16550_REG_SIZE < 0)
Simon Glassdd5497c2011-10-15 19:14:09 +000033#define serial_out(x, y) out_le32(y, x)
34#define serial_in(y) in_le32(y)
Graeme Russ14f06e62010-04-24 00:05:46 +100035#else
Simon Glassdd5497c2011-10-15 19:14:09 +000036#define serial_out(x, y) writeb(x, y)
37#define serial_in(y) readb(y)
Graeme Russ14f06e62010-04-24 00:05:46 +100038#endif
Simon Glass79a9da32014-09-04 16:27:34 -060039#endif /* !CONFIG_DM_SERIAL */
wdenke85390d2002-04-01 14:29:03 +000040
Khoronzhuk, Ivan80902982014-07-16 00:59:25 +030041#if defined(CONFIG_SOC_KEYSTONE)
Vitaly Andrianov7bcf4d62014-04-04 13:16:53 -040042#define UART_REG_VAL_PWREMU_MGMT_UART_DISABLE 0
43#define UART_REG_VAL_PWREMU_MGMT_UART_ENABLE ((1 << 14) | (1 << 13) | (1 << 0))
Karicheri, Muralidharancbc08882014-04-09 15:38:46 -040044#undef UART_MCRVAL
45#ifdef CONFIG_SERIAL_HW_FLOW_CONTROL
46#define UART_MCRVAL (UART_MCR_RTS | UART_MCR_AFE)
47#else
48#define UART_MCRVAL (UART_MCR_RTS)
49#endif
Vitaly Andrianov7bcf4d62014-04-04 13:16:53 -040050#endif
51
Prafulla Wadaskar66216372010-10-27 21:58:31 +053052#ifndef CONFIG_SYS_NS16550_IER
53#define CONFIG_SYS_NS16550_IER 0x00
54#endif /* CONFIG_SYS_NS16550_IER */
55
Simon Glassb31fb122015-02-27 22:06:26 -070056static inline void serial_out_shift(void *addr, int shift, int value)
Simon Glass6aba4fd2015-01-26 18:27:08 -070057{
Simon Glass79a9da32014-09-04 16:27:34 -060058#ifdef CONFIG_SYS_NS16550_PORT_MAPPED
Simon Glass96e230b2014-10-10 07:49:13 -060059 outb(value, (ulong)addr);
Bernhard Messerklingerd8427e72018-02-15 09:02:26 +010060#elif defined(CONFIG_SYS_NS16550_MEM32) && defined(CONFIG_SYS_LITTLE_ENDIAN)
Simon Glass79a9da32014-09-04 16:27:34 -060061 out_le32(addr, value);
62#elif defined(CONFIG_SYS_NS16550_MEM32) && defined(CONFIG_SYS_BIG_ENDIAN)
63 out_be32(addr, value);
Simon Glass0b31ec72015-05-12 14:55:02 -060064#elif defined(CONFIG_SYS_NS16550_MEM32)
65 writel(value, addr);
Simon Glass79a9da32014-09-04 16:27:34 -060066#elif defined(CONFIG_SYS_BIG_ENDIAN)
Simon Glass6aba4fd2015-01-26 18:27:08 -070067 writeb(value, addr + (1 << shift) - 1);
Simon Glass79a9da32014-09-04 16:27:34 -060068#else
69 writeb(value, addr);
70#endif
71}
72
Simon Glassb31fb122015-02-27 22:06:26 -070073static inline int serial_in_shift(void *addr, int shift)
Simon Glass79a9da32014-09-04 16:27:34 -060074{
Simon Glass79a9da32014-09-04 16:27:34 -060075#ifdef CONFIG_SYS_NS16550_PORT_MAPPED
Simon Glass96e230b2014-10-10 07:49:13 -060076 return inb((ulong)addr);
Bernhard Messerklingerd8427e72018-02-15 09:02:26 +010077#elif defined(CONFIG_SYS_NS16550_MEM32) && defined(CONFIG_SYS_LITTLE_ENDIAN)
Simon Glass79a9da32014-09-04 16:27:34 -060078 return in_le32(addr);
79#elif defined(CONFIG_SYS_NS16550_MEM32) && defined(CONFIG_SYS_BIG_ENDIAN)
80 return in_be32(addr);
Simon Glass0b31ec72015-05-12 14:55:02 -060081#elif defined(CONFIG_SYS_NS16550_MEM32)
82 return readl(addr);
Simon Glass79a9da32014-09-04 16:27:34 -060083#elif defined(CONFIG_SYS_BIG_ENDIAN)
Axel Linb5c372d2015-02-28 15:55:36 +080084 return readb(addr + (1 << shift) - 1);
Simon Glass79a9da32014-09-04 16:27:34 -060085#else
86 return readb(addr);
87#endif
88}
89
Simon Glassd945a492019-09-25 08:11:14 -060090#if CONFIG_IS_ENABLED(DM_SERIAL)
Marek Vasut3e97cbb2016-05-25 02:13:03 +020091
92#ifndef CONFIG_SYS_NS16550_CLK
93#define CONFIG_SYS_NS16550_CLK 0
94#endif
95
Simon Glass6aba4fd2015-01-26 18:27:08 -070096static void ns16550_writeb(NS16550_t port, int offset, int value)
97{
98 struct ns16550_platdata *plat = port->plat;
99 unsigned char *addr;
100
101 offset *= 1 << plat->reg_shift;
Paul Burtonc8acc892016-05-17 07:43:26 +0100102 addr = (unsigned char *)plat->base + offset;
103
Simon Glass6aba4fd2015-01-26 18:27:08 -0700104 /*
105 * As far as we know it doesn't make sense to support selection of
106 * these options at run-time, so use the existing CONFIG options.
107 */
Michal Simek7e0cdc42016-02-16 16:17:49 +0100108 serial_out_shift(addr + plat->reg_offset, plat->reg_shift, value);
Simon Glass6aba4fd2015-01-26 18:27:08 -0700109}
110
111static int ns16550_readb(NS16550_t port, int offset)
112{
113 struct ns16550_platdata *plat = port->plat;
114 unsigned char *addr;
115
116 offset *= 1 << plat->reg_shift;
Paul Burtonc8acc892016-05-17 07:43:26 +0100117 addr = (unsigned char *)plat->base + offset;
Simon Glass6aba4fd2015-01-26 18:27:08 -0700118
Michal Simek7e0cdc42016-02-16 16:17:49 +0100119 return serial_in_shift(addr + plat->reg_offset, plat->reg_shift);
Simon Glass6aba4fd2015-01-26 18:27:08 -0700120}
121
Marek Vasutf523c9c2016-12-01 02:06:29 +0100122static u32 ns16550_getfcr(NS16550_t port)
123{
124 struct ns16550_platdata *plat = port->plat;
125
126 return plat->fcr;
127}
128
Simon Glass79a9da32014-09-04 16:27:34 -0600129/* We can clean these up once everything is moved to driver model */
130#define serial_out(value, addr) \
Simon Glassb31fb122015-02-27 22:06:26 -0700131 ns16550_writeb(com_port, \
132 (unsigned char *)addr - (unsigned char *)com_port, value)
Simon Glass79a9da32014-09-04 16:27:34 -0600133#define serial_in(addr) \
Simon Glassb31fb122015-02-27 22:06:26 -0700134 ns16550_readb(com_port, \
135 (unsigned char *)addr - (unsigned char *)com_port)
Marek Vasutf523c9c2016-12-01 02:06:29 +0100136#else
137static u32 ns16550_getfcr(NS16550_t port)
138{
Heiko Schocher06f108e2017-01-18 08:05:49 +0100139 return UART_FCR_DEFVAL;
Marek Vasutf523c9c2016-12-01 02:06:29 +0100140}
Simon Glass79a9da32014-09-04 16:27:34 -0600141#endif
142
Marek Vasut3b164a52016-05-25 02:13:16 +0200143int ns16550_calc_divisor(NS16550_t port, int clock, int baudrate)
Simon Glasse98e01e2014-09-04 16:27:32 -0600144{
145 const unsigned int mode_x_div = 16;
146
Simon Glass27afb522015-01-26 18:27:09 -0700147 return DIV_ROUND_CLOSEST(clock, mode_x_div * baudrate);
148}
149
Simon Glassc31ebfe2014-09-04 16:27:33 -0600150static void NS16550_setbrg(NS16550_t com_port, int baud_divisor)
151{
Simon Goldschmidtba2d0aa2018-11-02 21:28:08 +0100152 /* to keep serial format, read lcr before writing BKSE */
153 int lcr_val = serial_in(&com_port->lcr) & ~UART_LCR_BKSE;
154
155 serial_out(UART_LCR_BKSE | lcr_val, &com_port->lcr);
Simon Glassc31ebfe2014-09-04 16:27:33 -0600156 serial_out(baud_divisor & 0xff, &com_port->dll);
157 serial_out((baud_divisor >> 8) & 0xff, &com_port->dlm);
Simon Goldschmidtba2d0aa2018-11-02 21:28:08 +0100158 serial_out(lcr_val, &com_port->lcr);
Simon Glassc31ebfe2014-09-04 16:27:33 -0600159}
160
Simon Glassdd5497c2011-10-15 19:14:09 +0000161void NS16550_init(NS16550_t com_port, int baud_divisor)
wdenke85390d2002-04-01 14:29:03 +0000162{
Gregoire Gentil6b05d0a2014-11-10 11:04:10 -0800163#if (defined(CONFIG_SPL_BUILD) && \
164 (defined(CONFIG_OMAP34XX) || defined(CONFIG_OMAP44XX)))
Manfred Huberf9b8ae32013-03-29 02:52:36 +0000165 /*
Gregoire Gentil6b05d0a2014-11-10 11:04:10 -0800166 * On some OMAP3/OMAP4 devices when UART3 is configured for boot mode
167 * before SPL starts only THRE bit is set. We have to empty the
168 * transmitter before initialization starts.
Manfred Huberf9b8ae32013-03-29 02:52:36 +0000169 */
170 if ((serial_in(&com_port->lsr) & (UART_LSR_TEMT | UART_LSR_THRE))
171 == UART_LSR_THRE) {
Simon Glass79a9da32014-09-04 16:27:34 -0600172 if (baud_divisor != -1)
173 NS16550_setbrg(com_port, baud_divisor);
Patrik Dahlström7e1bda92019-12-21 17:25:12 +0100174 else {
175 // Re-use old baud rate divisor to flush transmit reg.
176 const int dll = serial_in(&com_port->dll);
177 const int dlm = serial_in(&com_port->dlm);
178 const int divisor = dll | (dlm << 8);
179 NS16550_setbrg(com_port, divisor);
180 }
Manfred Huberf9b8ae32013-03-29 02:52:36 +0000181 serial_out(0, &com_port->mdr1);
182 }
183#endif
184
Scott Wood6c6f0612012-09-18 18:19:05 -0500185 while (!(serial_in(&com_port->lsr) & UART_LSR_TEMT))
186 ;
187
Prafulla Wadaskar66216372010-10-27 21:58:31 +0530188 serial_out(CONFIG_SYS_NS16550_IER, &com_port->ier);
Lokesh Vutla42cb4b82018-08-27 15:55:24 +0530189#if defined(CONFIG_ARCH_OMAP2PLUS) || defined(CONFIG_OMAP_SERIAL)
Graeme Russ14f06e62010-04-24 00:05:46 +1000190 serial_out(0x7, &com_port->mdr1); /* mode select reset TL16C750*/
wdenk21136db2003-07-16 21:53:01 +0000191#endif
Ley Foon Tana1763392018-06-14 18:45:22 +0800192
Graeme Russ14f06e62010-04-24 00:05:46 +1000193 serial_out(UART_MCRVAL, &com_port->mcr);
Marek Vasutf523c9c2016-12-01 02:06:29 +0100194 serial_out(ns16550_getfcr(com_port), &com_port->fcr);
Simon Goldschmidtba2d0aa2018-11-02 21:28:08 +0100195 /* initialize serial config to 8N1 before writing baudrate */
196 serial_out(UART_LCRVAL, &com_port->lcr);
Simon Glass79a9da32014-09-04 16:27:34 -0600197 if (baud_divisor != -1)
198 NS16550_setbrg(com_port, baud_divisor);
Lokesh Vutla42cb4b82018-08-27 15:55:24 +0530199#if defined(CONFIG_ARCH_OMAP2PLUS) || defined(CONFIG_SOC_DA8XX) || \
200 defined(CONFIG_OMAP_SERIAL)
Simon Glassdd5497c2011-10-15 19:14:09 +0000201 /* /16 is proper to hit 115200 with 48MHz */
202 serial_out(0, &com_port->mdr1);
Tom Rinif28c4342017-05-12 22:33:16 -0400203#endif
Khoronzhuk, Ivan80902982014-07-16 00:59:25 +0300204#if defined(CONFIG_SOC_KEYSTONE)
Vitaly Andrianov7bcf4d62014-04-04 13:16:53 -0400205 serial_out(UART_REG_VAL_PWREMU_MGMT_UART_ENABLE, &com_port->regC);
206#endif
wdenke85390d2002-04-01 14:29:03 +0000207}
208
Ron Madriddfa028a2009-02-18 14:30:44 -0800209#ifndef CONFIG_NS16550_MIN_FUNCTIONS
Simon Glassdd5497c2011-10-15 19:14:09 +0000210void NS16550_reinit(NS16550_t com_port, int baud_divisor)
wdenke85390d2002-04-01 14:29:03 +0000211{
Prafulla Wadaskar66216372010-10-27 21:58:31 +0530212 serial_out(CONFIG_SYS_NS16550_IER, &com_port->ier);
Simon Glassc31ebfe2014-09-04 16:27:33 -0600213 NS16550_setbrg(com_port, 0);
Graeme Russ14f06e62010-04-24 00:05:46 +1000214 serial_out(UART_MCRVAL, &com_port->mcr);
Marek Vasutf523c9c2016-12-01 02:06:29 +0100215 serial_out(ns16550_getfcr(com_port), &com_port->fcr);
Simon Glassc31ebfe2014-09-04 16:27:33 -0600216 NS16550_setbrg(com_port, baud_divisor);
wdenke85390d2002-04-01 14:29:03 +0000217}
Ron Madriddfa028a2009-02-18 14:30:44 -0800218#endif /* CONFIG_NS16550_MIN_FUNCTIONS */
wdenke85390d2002-04-01 14:29:03 +0000219
Simon Glassdd5497c2011-10-15 19:14:09 +0000220void NS16550_putc(NS16550_t com_port, char c)
wdenke85390d2002-04-01 14:29:03 +0000221{
Simon Glassdd5497c2011-10-15 19:14:09 +0000222 while ((serial_in(&com_port->lsr) & UART_LSR_THRE) == 0)
223 ;
Graeme Russ14f06e62010-04-24 00:05:46 +1000224 serial_out(c, &com_port->thr);
Stefan Roese57b99882010-10-12 09:39:45 +0200225
226 /*
227 * Call watchdog_reset() upon newline. This is done here in putc
228 * since the environment code uses a single puts() to print the complete
229 * environment upon "printenv". So we can't put this watchdog call
230 * in puts().
231 */
232 if (c == '\n')
233 WATCHDOG_RESET();
wdenke85390d2002-04-01 14:29:03 +0000234}
235
Ron Madriddfa028a2009-02-18 14:30:44 -0800236#ifndef CONFIG_NS16550_MIN_FUNCTIONS
Simon Glassdd5497c2011-10-15 19:14:09 +0000237char NS16550_getc(NS16550_t com_port)
wdenke85390d2002-04-01 14:29:03 +0000238{
Graeme Russ14f06e62010-04-24 00:05:46 +1000239 while ((serial_in(&com_port->lsr) & UART_LSR_DR) == 0) {
Marek Vasut9e1fca92012-09-15 10:25:19 +0200240#if !defined(CONFIG_SPL_BUILD) && defined(CONFIG_USB_TTY)
wdenk29e7f5a2004-03-12 00:14:09 +0000241 extern void usbtty_poll(void);
242 usbtty_poll();
243#endif
Ladislav Michlcc294422010-02-01 23:34:25 +0100244 WATCHDOG_RESET();
wdenk29e7f5a2004-03-12 00:14:09 +0000245 }
Graeme Russ14f06e62010-04-24 00:05:46 +1000246 return serial_in(&com_port->rbr);
wdenke85390d2002-04-01 14:29:03 +0000247}
248
Simon Glassdd5497c2011-10-15 19:14:09 +0000249int NS16550_tstc(NS16550_t com_port)
wdenke85390d2002-04-01 14:29:03 +0000250{
Simon Glassdd5497c2011-10-15 19:14:09 +0000251 return (serial_in(&com_port->lsr) & UART_LSR_DR) != 0;
wdenke85390d2002-04-01 14:29:03 +0000252}
253
Ron Madriddfa028a2009-02-18 14:30:44 -0800254#endif /* CONFIG_NS16550_MIN_FUNCTIONS */
Simon Glass79a9da32014-09-04 16:27:34 -0600255
Simon Glass27afb522015-01-26 18:27:09 -0700256#ifdef CONFIG_DEBUG_UART_NS16550
257
258#include <debug_uart.h>
259
Simon Glass60517d72015-10-18 19:51:23 -0600260static inline void _debug_uart_init(void)
Simon Glass27afb522015-01-26 18:27:09 -0700261{
262 struct NS16550 *com_port = (struct NS16550 *)CONFIG_DEBUG_UART_BASE;
263 int baud_divisor;
264
265 /*
266 * We copy the code from above because it is already horribly messy.
267 * Trying to refactor to nicely remove the duplication doesn't seem
268 * feasible. The better fix is to move all users of this driver to
269 * driver model.
270 */
Marek Vasut3b164a52016-05-25 02:13:16 +0200271 baud_divisor = ns16550_calc_divisor(com_port, CONFIG_DEBUG_UART_CLOCK,
272 CONFIG_BAUDRATE);
Simon Glass92465162015-06-23 15:39:06 -0600273 serial_dout(&com_port->ier, CONFIG_SYS_NS16550_IER);
Lokesh Vutla771d69c2017-04-22 15:57:25 +0530274 serial_dout(&com_port->mcr, UART_MCRVAL);
275 serial_dout(&com_port->fcr, UART_FCR_DEFVAL);
276
277 serial_dout(&com_port->lcr, UART_LCR_BKSE | UART_LCRVAL);
278 serial_dout(&com_port->dll, baud_divisor & 0xff);
279 serial_dout(&com_port->dlm, (baud_divisor >> 8) & 0xff);
280 serial_dout(&com_port->lcr, UART_LCRVAL);
281}
282
Simon Goldschmidte03ad212019-01-09 20:35:31 +0100283static inline int NS16550_read_baud_divisor(struct NS16550 *com_port)
284{
285 int ret;
286
287 serial_dout(&com_port->lcr, UART_LCR_BKSE | UART_LCRVAL);
288 ret = serial_din(&com_port->dll) & 0xff;
289 ret |= (serial_din(&com_port->dlm) & 0xff) << 8;
290 serial_dout(&com_port->lcr, UART_LCRVAL);
291
292 return ret;
293}
294
Lokesh Vutla771d69c2017-04-22 15:57:25 +0530295static inline void _debug_uart_putc(int ch)
296{
297 struct NS16550 *com_port = (struct NS16550 *)CONFIG_DEBUG_UART_BASE;
298
Simon Goldschmidte03ad212019-01-09 20:35:31 +0100299 while (!(serial_din(&com_port->lsr) & UART_LSR_THRE)) {
300#ifdef CONFIG_DEBUG_UART_NS16550_CHECK_ENABLED
301 if (!NS16550_read_baud_divisor(com_port))
302 return;
303#endif
304 }
Lokesh Vutla771d69c2017-04-22 15:57:25 +0530305 serial_dout(&com_port->thr, ch);
306}
307
308DEBUG_UART_FUNCS
309
310#endif
311
Simon Glassd945a492019-09-25 08:11:14 -0600312#if CONFIG_IS_ENABLED(DM_SERIAL)
Simon Glass79a9da32014-09-04 16:27:34 -0600313static int ns16550_serial_putc(struct udevice *dev, const char ch)
314{
315 struct NS16550 *const com_port = dev_get_priv(dev);
316
317 if (!(serial_in(&com_port->lsr) & UART_LSR_THRE))
318 return -EAGAIN;
319 serial_out(ch, &com_port->thr);
320
321 /*
322 * Call watchdog_reset() upon newline. This is done here in putc
323 * since the environment code uses a single puts() to print the complete
324 * environment upon "printenv". So we can't put this watchdog call
325 * in puts().
326 */
327 if (ch == '\n')
328 WATCHDOG_RESET();
329
330 return 0;
331}
332
333static int ns16550_serial_pending(struct udevice *dev, bool input)
334{
335 struct NS16550 *const com_port = dev_get_priv(dev);
336
337 if (input)
Mario Sixaed664a2018-01-15 11:09:49 +0100338 return (serial_in(&com_port->lsr) & UART_LSR_DR) ? 1 : 0;
Simon Glass79a9da32014-09-04 16:27:34 -0600339 else
Mario Sixaed664a2018-01-15 11:09:49 +0100340 return (serial_in(&com_port->lsr) & UART_LSR_THRE) ? 0 : 1;
Simon Glass79a9da32014-09-04 16:27:34 -0600341}
342
343static int ns16550_serial_getc(struct udevice *dev)
344{
Stefan Roese48b771c2017-08-16 17:37:15 +0200345 struct NS16550 *const com_port = dev_get_priv(dev);
346
347 if (!(serial_in(&com_port->lsr) & UART_LSR_DR))
Simon Glass79a9da32014-09-04 16:27:34 -0600348 return -EAGAIN;
349
Stefan Roese48b771c2017-08-16 17:37:15 +0200350 return serial_in(&com_port->rbr);
Simon Glass79a9da32014-09-04 16:27:34 -0600351}
352
353static int ns16550_serial_setbrg(struct udevice *dev, int baudrate)
354{
355 struct NS16550 *const com_port = dev_get_priv(dev);
356 struct ns16550_platdata *plat = com_port->plat;
357 int clock_divisor;
358
359 clock_divisor = ns16550_calc_divisor(com_port, plat->clock, baudrate);
360
361 NS16550_setbrg(com_port, clock_divisor);
362
Simon Goldschmidtba2d0aa2018-11-02 21:28:08 +0100363 return 0;
364}
365
366static int ns16550_serial_setconfig(struct udevice *dev, uint serial_config)
367{
368 struct NS16550 *const com_port = dev_get_priv(dev);
369 int lcr_val = UART_LCR_WLS_8;
370 uint parity = SERIAL_GET_PARITY(serial_config);
371 uint bits = SERIAL_GET_BITS(serial_config);
372 uint stop = SERIAL_GET_STOP(serial_config);
373
374 /*
375 * only parity config is implemented, check if other serial settings
376 * are the default one.
377 */
378 if (bits != SERIAL_8_BITS || stop != SERIAL_ONE_STOP)
379 return -ENOTSUPP; /* not supported in driver*/
380
381 switch (parity) {
382 case SERIAL_PAR_NONE:
383 /* no bits to add */
384 break;
385 case SERIAL_PAR_ODD:
386 lcr_val |= UART_LCR_PEN;
387 break;
388 case SERIAL_PAR_EVEN:
389 lcr_val |= UART_LCR_PEN | UART_LCR_EPS;
390 break;
391 default:
392 return -ENOTSUPP; /* not supported in driver*/
393 }
394
395 serial_out(lcr_val, &com_port->lcr);
Simon Glass79a9da32014-09-04 16:27:34 -0600396 return 0;
397}
398
Andy Shevchenkod778fc42018-11-20 23:52:36 +0200399static int ns16550_serial_getinfo(struct udevice *dev,
400 struct serial_device_info *info)
401{
402 struct NS16550 *const com_port = dev_get_priv(dev);
403 struct ns16550_platdata *plat = com_port->plat;
404
405 info->type = SERIAL_CHIP_16550_COMPATIBLE;
406#ifdef CONFIG_SYS_NS16550_PORT_MAPPED
407 info->addr_space = SERIAL_ADDRESS_SPACE_IO;
408#else
409 info->addr_space = SERIAL_ADDRESS_SPACE_MEMORY;
410#endif
411 info->addr = plat->base;
412 info->reg_width = plat->reg_width;
413 info->reg_shift = plat->reg_shift;
414 info->reg_offset = plat->reg_offset;
415 return 0;
416}
417
Simon Glass79a9da32014-09-04 16:27:34 -0600418int ns16550_serial_probe(struct udevice *dev)
419{
420 struct NS16550 *const com_port = dev_get_priv(dev);
Ley Foon Tana1763392018-06-14 18:45:22 +0800421 struct reset_ctl_bulk reset_bulk;
422 int ret;
423
424 ret = reset_get_bulk(dev, &reset_bulk);
425 if (!ret)
426 reset_deassert_bulk(&reset_bulk);
Simon Glass79a9da32014-09-04 16:27:34 -0600427
Simon Glass3bf04f32014-10-22 21:37:05 -0600428 com_port->plat = dev_get_platdata(dev);
Simon Glass79a9da32014-09-04 16:27:34 -0600429 NS16550_init(com_port, -1);
430
431 return 0;
432}
433
Marek Vasut1a59eec2016-12-01 02:06:30 +0100434#if CONFIG_IS_ENABLED(OF_CONTROL)
435enum {
436 PORT_NS16550 = 0,
Marek Vasut92a744f2016-12-01 02:06:31 +0100437 PORT_JZ4780,
Marek Vasut1a59eec2016-12-01 02:06:30 +0100438};
439#endif
440
Simon Glass18798be2016-07-04 11:58:23 -0600441#if CONFIG_IS_ENABLED(OF_CONTROL) && !CONFIG_IS_ENABLED(OF_PLATDATA)
Simon Glass79a9da32014-09-04 16:27:34 -0600442int ns16550_serial_ofdata_to_platdata(struct udevice *dev)
443{
Simon Glass79a9da32014-09-04 16:27:34 -0600444 struct ns16550_platdata *plat = dev->platdata;
Marek Vasut92a744f2016-12-01 02:06:31 +0100445 const u32 port_type = dev_get_driver_data(dev);
Simon Glass79a9da32014-09-04 16:27:34 -0600446 fdt_addr_t addr;
Masahiro Yamada09abe2b2016-09-26 20:45:27 +0900447 struct clk clk;
448 int err;
Simon Glass79a9da32014-09-04 16:27:34 -0600449
Bin Meng0203c1c2014-12-31 16:05:12 +0800450 /* try Processor Local Bus device first */
Simon Glass23b27592019-09-15 12:08:58 -0600451 addr = dev_read_addr_pci(dev);
Simon Glass79a9da32014-09-04 16:27:34 -0600452 if (addr == FDT_ADDR_T_NONE)
453 return -EINVAL;
454
Paul Burtonc8acc892016-05-17 07:43:26 +0100455#ifdef CONFIG_SYS_NS16550_PORT_MAPPED
Simon Glass25463942014-10-22 21:37:04 -0600456 plat->base = addr;
Paul Burtonc8acc892016-05-17 07:43:26 +0100457#else
458 plat->base = (unsigned long)map_physmem(addr, 0, MAP_NOCACHE);
459#endif
460
Philipp Tomsiche74958e2017-06-07 18:46:02 +0200461 plat->reg_offset = dev_read_u32_default(dev, "reg-offset", 0);
462 plat->reg_shift = dev_read_u32_default(dev, "reg-shift", 0);
Andy Shevchenko72fccfe2018-11-20 23:52:35 +0200463 plat->reg_width = dev_read_u32_default(dev, "reg-io-width", 1);
Paul Burton8aadc562016-09-08 07:47:29 +0100464
465 err = clk_get_by_index(dev, 0, &clk);
466 if (!err) {
467 err = clk_get_rate(&clk);
468 if (!IS_ERR_VALUE(err))
469 plat->clock = err;
Alexandre Courboteaa24e7f2016-09-30 17:37:00 +0900470 } else if (err != -ENOENT && err != -ENODEV && err != -ENOSYS) {
Paul Burton8aadc562016-09-08 07:47:29 +0100471 debug("ns16550 failed to get clock\n");
472 return err;
473 }
474
475 if (!plat->clock)
Philipp Tomsiche74958e2017-06-07 18:46:02 +0200476 plat->clock = dev_read_u32_default(dev, "clock-frequency",
477 CONFIG_SYS_NS16550_CLK);
Thomas Chou16b2e7e2015-11-19 21:48:05 +0800478 if (!plat->clock) {
479 debug("ns16550 clock not defined\n");
480 return -EINVAL;
481 }
Simon Glass79a9da32014-09-04 16:27:34 -0600482
Heiko Schocher06f108e2017-01-18 08:05:49 +0100483 plat->fcr = UART_FCR_DEFVAL;
Marek Vasut92a744f2016-12-01 02:06:31 +0100484 if (port_type == PORT_JZ4780)
485 plat->fcr |= UART_FCR_UME;
Marek Vasutf523c9c2016-12-01 02:06:29 +0100486
Simon Glass79a9da32014-09-04 16:27:34 -0600487 return 0;
488}
Simon Glass3bf04f32014-10-22 21:37:05 -0600489#endif
Simon Glass79a9da32014-09-04 16:27:34 -0600490
491const struct dm_serial_ops ns16550_serial_ops = {
492 .putc = ns16550_serial_putc,
493 .pending = ns16550_serial_pending,
494 .getc = ns16550_serial_getc,
495 .setbrg = ns16550_serial_setbrg,
Andy Shevchenkod778fc42018-11-20 23:52:36 +0200496 .setconfig = ns16550_serial_setconfig,
497 .getinfo = ns16550_serial_getinfo,
Simon Glass79a9da32014-09-04 16:27:34 -0600498};
Thomas Chou16b2e7e2015-11-19 21:48:05 +0800499
Alexandru Gagniucb1ab1892017-03-27 12:54:19 -0700500#if CONFIG_IS_ENABLED(OF_CONTROL) && !CONFIG_IS_ENABLED(OF_PLATDATA)
Thomas Choudad53672015-12-14 20:45:09 +0800501/*
502 * Please consider existing compatible strings before adding a new
503 * one to keep this table compact. Or you may add a generic "ns16550"
504 * compatible string to your dts.
505 */
Thomas Chou16b2e7e2015-11-19 21:48:05 +0800506static const struct udevice_id ns16550_serial_ids[] = {
Marek Vasut1a59eec2016-12-01 02:06:30 +0100507 { .compatible = "ns16550", .data = PORT_NS16550 },
508 { .compatible = "ns16550a", .data = PORT_NS16550 },
Marek Vasut92a744f2016-12-01 02:06:31 +0100509 { .compatible = "ingenic,jz4780-uart", .data = PORT_JZ4780 },
Marek Vasut1a59eec2016-12-01 02:06:30 +0100510 { .compatible = "nvidia,tegra20-uart", .data = PORT_NS16550 },
511 { .compatible = "snps,dw-apb-uart", .data = PORT_NS16550 },
Thomas Chou16b2e7e2015-11-19 21:48:05 +0800512 {}
513};
Alexandru Gagniucb1ab1892017-03-27 12:54:19 -0700514#endif /* OF_CONTROL && !OF_PLATDATA */
Thomas Chou16b2e7e2015-11-19 21:48:05 +0800515
Simon Glass9ebf3482015-12-13 21:36:59 -0700516#if CONFIG_IS_ENABLED(SERIAL_PRESENT)
Alexandru Gagniucb1ab1892017-03-27 12:54:19 -0700517
518/* TODO(sjg@chromium.org): Integrate this into a macro like CONFIG_IS_ENABLED */
519#if !defined(CONFIG_TPL_BUILD) || defined(CONFIG_TPL_DM_SERIAL)
Thomas Chou16b2e7e2015-11-19 21:48:05 +0800520U_BOOT_DRIVER(ns16550_serial) = {
521 .name = "ns16550_serial",
522 .id = UCLASS_SERIAL,
Alexandru Gagniucb1ab1892017-03-27 12:54:19 -0700523#if CONFIG_IS_ENABLED(OF_CONTROL) && !CONFIG_IS_ENABLED(OF_PLATDATA)
Thomas Chou16b2e7e2015-11-19 21:48:05 +0800524 .of_match = ns16550_serial_ids,
525 .ofdata_to_platdata = ns16550_serial_ofdata_to_platdata,
526 .platdata_auto_alloc_size = sizeof(struct ns16550_platdata),
527#endif
528 .priv_auto_alloc_size = sizeof(struct NS16550),
529 .probe = ns16550_serial_probe,
530 .ops = &ns16550_serial_ops,
Bin Mengbdb33d82018-10-24 06:36:36 -0700531#if !CONFIG_IS_ENABLED(OF_CONTROL)
Simon Glass6ef533e2015-12-04 08:58:38 -0700532 .flags = DM_FLAG_PRE_RELOC,
Bin Mengbdb33d82018-10-24 06:36:36 -0700533#endif
Thomas Chou16b2e7e2015-11-19 21:48:05 +0800534};
Simon Glass9ebf3482015-12-13 21:36:59 -0700535#endif
Alexandru Gagniucb1ab1892017-03-27 12:54:19 -0700536#endif /* SERIAL_PRESENT */
537
Simon Glass79a9da32014-09-04 16:27:34 -0600538#endif /* CONFIG_DM_SERIAL */