xref: /linux/drivers/tty/serial/earlycon-riscv-sbi.c (revision 06d07429858317ded2db7986113a9e0129cd599b)
127de1f54SAnup Patel // SPDX-License-Identifier: GPL-2.0
227de1f54SAnup Patel /*
327de1f54SAnup Patel  * RISC-V SBI based earlycon
427de1f54SAnup Patel  *
527de1f54SAnup Patel  * Copyright (C) 2018 Anup Patel <anup@brainfault.org>
627de1f54SAnup Patel  */
727de1f54SAnup Patel #include <linux/kernel.h>
827de1f54SAnup Patel #include <linux/console.h>
927de1f54SAnup Patel #include <linux/init.h>
1027de1f54SAnup Patel #include <linux/serial_core.h>
1127de1f54SAnup Patel #include <asm/sbi.h>
1227de1f54SAnup Patel 
sbi_putc(struct uart_port * port,unsigned char c)133f8bab17SJiri Slaby static void sbi_putc(struct uart_port *port, unsigned char c)
1427de1f54SAnup Patel {
15a19f7470SAndreas Schwab 	sbi_console_putchar(c);
16a19f7470SAndreas Schwab }
1727de1f54SAnup Patel 
sbi_0_1_console_write(struct console * con,const char * s,unsigned int n)18*c77bf360SAnup Patel static void sbi_0_1_console_write(struct console *con,
19*c77bf360SAnup Patel 				  const char *s, unsigned int n)
20a19f7470SAndreas Schwab {
21a19f7470SAndreas Schwab 	struct earlycon_device *dev = con->data;
22a19f7470SAndreas Schwab 	uart_console_write(&dev->port, s, n, sbi_putc);
2327de1f54SAnup Patel }
2427de1f54SAnup Patel 
sbi_dbcn_console_write(struct console * con,const char * s,unsigned int n)25*c77bf360SAnup Patel static void sbi_dbcn_console_write(struct console *con,
26*c77bf360SAnup Patel 				   const char *s, unsigned int n)
27*c77bf360SAnup Patel {
28*c77bf360SAnup Patel 	int ret;
29*c77bf360SAnup Patel 
30*c77bf360SAnup Patel 	while (n) {
31*c77bf360SAnup Patel 		ret = sbi_debug_console_write(s, n);
32*c77bf360SAnup Patel 		if (ret < 0)
33*c77bf360SAnup Patel 			break;
34*c77bf360SAnup Patel 
35*c77bf360SAnup Patel 		s += ret;
36*c77bf360SAnup Patel 		n -= ret;
37*c77bf360SAnup Patel 	}
38*c77bf360SAnup Patel }
39*c77bf360SAnup Patel 
early_sbi_setup(struct earlycon_device * device,const char * opt)4027de1f54SAnup Patel static int __init early_sbi_setup(struct earlycon_device *device,
4127de1f54SAnup Patel 				  const char *opt)
4227de1f54SAnup Patel {
43*c77bf360SAnup Patel 	if (sbi_debug_console_available)
44*c77bf360SAnup Patel 		device->con->write = sbi_dbcn_console_write;
45*c77bf360SAnup Patel 	else if (IS_ENABLED(CONFIG_RISCV_SBI_V01))
46*c77bf360SAnup Patel 		device->con->write = sbi_0_1_console_write;
47*c77bf360SAnup Patel 	else
48*c77bf360SAnup Patel 		return -ENODEV;
49*c77bf360SAnup Patel 
5027de1f54SAnup Patel 	return 0;
5127de1f54SAnup Patel }
5227de1f54SAnup Patel EARLYCON_DECLARE(sbi, early_sbi_setup);
53