mirror of
https://github.com/merbanan/airoha_ml.git
synced 2025-11-10 17:46:10 +00:00
236 lines
8.2 KiB
Plaintext
236 lines
8.2 KiB
Plaintext
From 535a2a494046b65be3a0cf6f19fdac9d21015e3e Mon Sep 17 00:00:00 2001
|
|
From: Markus Gothe <markus.gothe@genexis.eu>
|
|
Date: Wed, 11 Jan 2023 17:41:35 +0000
|
|
Subject: 8250: Add Airoha UART driver.
|
|
|
|
|
|
diff --git a/drivers/tty/serial/8250/8250_en7523.c b/drivers/tty/serial/8250/8250_en7523.c
|
|
new file mode 100644
|
|
index 000000000..dfee70767
|
|
--- /dev/null
|
|
+++ b/drivers/tty/serial/8250/8250_en7523.c
|
|
@@ -0,0 +1,94 @@
|
|
+// SPDX-License-Identifier: GPL-2.0+
|
|
+/*
|
|
+ * Airoha EN7523 driver.
|
|
+ *
|
|
+ * Copyright (c) 2022 Genexis Sweden AB
|
|
+ * Author: Benjamin Larsson <benjamin.larsson@genexis.eu>
|
|
+ */
|
|
+#include <linux/clk.h>
|
|
+#include <linux/io.h>
|
|
+#include <linux/module.h>
|
|
+#include <linux/of_irq.h>
|
|
+#include <linux/of_platform.h>
|
|
+#include <linux/pinctrl/consumer.h>
|
|
+#include <linux/platform_device.h>
|
|
+#include <linux/pm_runtime.h>
|
|
+#include <linux/serial_8250.h>
|
|
+#include <linux/serial_reg.h>
|
|
+#include <linux/console.h>
|
|
+#include <linux/dma-mapping.h>
|
|
+#include <linux/tty.h>
|
|
+#include <linux/tty_flip.h>
|
|
+
|
|
+#include "8250.h"
|
|
+
|
|
+
|
|
+/* The Airoha UART is 16550-compatible except for the baud rate calculation.
|
|
+ *
|
|
+ * crystal_clock = 20 MHz
|
|
+ * xindiv_clock = crystal_clock / clock_div
|
|
+ * (x/y) = XYD, 32 bit register with 16 bits of x and and then 16 bits of y
|
|
+ * clock_div = XINCLK_DIVCNT (default set to 10 (0x4)),
|
|
+ * - 3 bit register [ 1, 2, 4, 8, 10, 12, 16, 20 ]
|
|
+ *
|
|
+ * baud_rate = ((xindiv_clock) * (x/y)) / ([BRDH,BRDL] * 16)
|
|
+ *
|
|
+ * XYD_y seems to need to be larger then XYD_x for things to work.
|
|
+ * Setting [BRDH,BRDL] to [0,1] and XYD_y to 65000 give even values
|
|
+ * for usual baud rates.
|
|
+ *
|
|
+ * Selecting divider needs to fulfill
|
|
+ * 1.8432 MHz <= xindiv_clk <= APB clock / 2
|
|
+ * The clocks are unknown but a divider of value 1 did not work.
|
|
+ *
|
|
+ * Optimally the XYD, BRD and XINCLK_DIVCNT registers could be searched to
|
|
+ * find values that gives the least error for every baud rate. But searching
|
|
+ * the space takes time and in practise only a few rates are of interest.
|
|
+ * With some value combinations not working a tested subset is used giving
|
|
+ * a usable range from 110 to 460800 baud.
|
|
+ */
|
|
+
|
|
+#define CLOCK_DIV_TAB_ELEMS 3
|
|
+#define XYD_Y 65000
|
|
+#define XINDIV_CLOCK 20000000
|
|
+#define UART_BRDL_20M 0x01
|
|
+#define UART_BRDH_20M 0x00
|
|
+
|
|
+static int clock_div_tab[] = { 10, 4, 2};
|
|
+static int clock_div_reg[] = { 4, 2, 1};
|
|
+
|
|
+
|
|
+int en7523_set_uart_baud_rate (struct uart_port *port, unsigned int baud)
|
|
+{
|
|
+ struct uart_8250_port *up = up_to_u8250p(port);
|
|
+ unsigned int xyd_x, nom, denom;
|
|
+ int i;
|
|
+
|
|
+ /* set DLAB to access the baud rate divider registers (BRDH, BRDL) */
|
|
+ serial_port_out(port, UART_LCR, up->lcr | UART_LCR_DLAB);
|
|
+
|
|
+ /* set baud rate calculation defaults */
|
|
+
|
|
+ /* set BRDIV ([BRDH,BRDL]) to 1 */
|
|
+ serial_port_out(port, UART_BRDL, UART_BRDL_20M);
|
|
+ serial_port_out(port, UART_BRDH, UART_BRDH_20M);
|
|
+
|
|
+ /* calculate XYD_x and XINCLKDR register */
|
|
+
|
|
+ for (i = 0 ; i < CLOCK_DIV_TAB_ELEMS ; i++) {
|
|
+ denom = (XINDIV_CLOCK/40) / clock_div_tab[i];
|
|
+ nom = (baud * (XYD_Y/40));
|
|
+ xyd_x = ((nom/denom) << 4);
|
|
+ if (xyd_x < XYD_Y) break;
|
|
+ }
|
|
+
|
|
+ serial_port_out(port, UART_XINCLKDR, clock_div_reg[i]);
|
|
+ serial_port_out(port, UART_XYD, (xyd_x<<16) | XYD_Y);
|
|
+
|
|
+ /* unset DLAB */
|
|
+ serial_port_out(port, UART_LCR, up->lcr);
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+EXPORT_SYMBOL_GPL(en7523_set_uart_baud_rate);
|
|
diff --git a/drivers/tty/serial/8250/8250_of.c b/drivers/tty/serial/8250/8250_of.c
|
|
index 9ba31701a..cb07871ac 100644
|
|
--- a/drivers/tty/serial/8250/8250_of.c
|
|
+++ b/drivers/tty/serial/8250/8250_of.c
|
|
@@ -341,6 +341,7 @@ static const struct of_device_id of_platform_serial_table[] = {
|
|
.data = (void *)PORT_XSCALE, },
|
|
{ .compatible = "ti,da830-uart", .data = (void *)PORT_DA830, },
|
|
{ .compatible = "nuvoton,npcm750-uart", .data = (void *)PORT_NPCM, },
|
|
+ { .compatible = "airoha,en7523-uart", .data = (void *)PORT_AIROHA, },
|
|
{ /* end of list */ },
|
|
};
|
|
MODULE_DEVICE_TABLE(of, of_platform_serial_table);
|
|
diff --git a/drivers/tty/serial/8250/8250_port.c b/drivers/tty/serial/8250/8250_port.c
|
|
index dbb27303a..13009ace1 100644
|
|
--- a/drivers/tty/serial/8250/8250_port.c
|
|
+++ b/drivers/tty/serial/8250/8250_port.c
|
|
@@ -309,6 +309,11 @@ static const struct serial8250_config uart_config[] = {
|
|
.rxtrig_bytes = {1, 32, 64, 112},
|
|
.flags = UART_CAP_FIFO | UART_CAP_SLEEP,
|
|
},
|
|
+ [PORT_AIROHA] = {
|
|
+ .name = "Airoha 16550",
|
|
+ .fifo_size = 1,
|
|
+ .tx_loadsz = 1,
|
|
+ },
|
|
};
|
|
|
|
/* Uart divisor latch read */
|
|
@@ -2666,6 +2671,13 @@ serial8250_do_set_termios(struct uart_port *port, struct ktermios *termios,
|
|
|
|
serial8250_set_divisor(port, baud, quot, frac);
|
|
|
|
+
|
|
+ /*
|
|
+ * Airoha SoCs have custom registers for baud rate settings
|
|
+ */
|
|
+ if (port->type == PORT_AIROHA)
|
|
+ en7523_set_uart_baud_rate(port, baud);
|
|
+
|
|
/*
|
|
* LCR DLAB must be set to enable 64-byte FIFO mode. If the FCR
|
|
* is written without DLAB set, this mode will be disabled.
|
|
diff --git a/drivers/tty/serial/8250/Kconfig b/drivers/tty/serial/8250/Kconfig
|
|
index 7ef60f8b6..82de43f66 100644
|
|
--- a/drivers/tty/serial/8250/Kconfig
|
|
+++ b/drivers/tty/serial/8250/Kconfig
|
|
@@ -478,6 +478,10 @@ config SERIAL_8250_PXA
|
|
applicable to both devicetree and legacy boards, and early console is
|
|
part of its support.
|
|
|
|
+config SERIAL_8250_AIROHA
|
|
+ tristate "Airoha serial port support"
|
|
+ depends on SERIAL_8250
|
|
+
|
|
config SERIAL_OF_PLATFORM
|
|
tristate "Devicetree based probing for 8250 ports"
|
|
depends on SERIAL_8250 && OF
|
|
diff --git a/drivers/tty/serial/8250/Makefile b/drivers/tty/serial/8250/Makefile
|
|
index 08c1d8117..e3a3c5169 100644
|
|
--- a/drivers/tty/serial/8250/Makefile
|
|
+++ b/drivers/tty/serial/8250/Makefile
|
|
@@ -36,6 +36,7 @@ obj-$(CONFIG_SERIAL_8250_INGENIC) += 8250_ingenic.o
|
|
obj-$(CONFIG_SERIAL_8250_LPSS) += 8250_lpss.o
|
|
obj-$(CONFIG_SERIAL_8250_MID) += 8250_mid.o
|
|
obj-$(CONFIG_SERIAL_8250_PXA) += 8250_pxa.o
|
|
+obj-$(CONFIG_SERIAL_8250_AIROHA) += 8250_en7523.o
|
|
obj-$(CONFIG_SERIAL_OF_PLATFORM) += 8250_of.o
|
|
|
|
CFLAGS_8250_ingenic.o += -I$(srctree)/scripts/dtc/libfdt
|
|
diff --git a/drivers/tty/serial/Makefile b/drivers/tty/serial/Makefile
|
|
index 3d911a3e2..f1a5d8420 100644
|
|
--- a/drivers/tty/serial/Makefile
|
|
+++ b/drivers/tty/serial/Makefile
|
|
@@ -65,10 +65,10 @@ obj-$(CONFIG_SERIAL_GRLIB_GAISLER_APBUART) += apbuart.o
|
|
obj-$(CONFIG_SERIAL_ALTERA_JTAGUART) += altera_jtaguart.o
|
|
obj-$(CONFIG_SERIAL_VT8500) += vt8500_serial.o
|
|
ifeq ($(strip $(TCSUPPORT_KERNEL_API)),)
|
|
-obj-y += tc3162_uart.o
|
|
+#obj-y += tc3162_uart.o
|
|
endif
|
|
ifeq ($(TCSUPPORT_UART2),1)
|
|
-obj-y += tc3162_uart2.o
|
|
+#obj-y += tc3162_uart2.o
|
|
endif
|
|
obj-$(CONFIG_SERIAL_IFX6X60) += ifx6x60.o
|
|
obj-$(CONFIG_SERIAL_PCH_UART) += pch_uart.o
|
|
diff --git a/include/linux/serial_8250.h b/include/linux/serial_8250.h
|
|
index bb2bc9938..df24d293b 100644
|
|
--- a/include/linux/serial_8250.h
|
|
+++ b/include/linux/serial_8250.h
|
|
@@ -165,6 +165,7 @@ extern void serial8250_do_set_divisor(struct uart_port *port, unsigned int baud,
|
|
unsigned int quot,
|
|
unsigned int quot_frac);
|
|
extern int fsl8250_handle_irq(struct uart_port *port);
|
|
+extern int en7523_set_uart_baud_rate(struct uart_port *port, unsigned int baud);
|
|
int serial8250_handle_irq(struct uart_port *port, unsigned int iir);
|
|
unsigned char serial8250_rx_chars(struct uart_8250_port *up, unsigned char lsr);
|
|
void serial8250_read_char(struct uart_8250_port *up, unsigned char lsr);
|
|
diff --git a/include/uapi/linux/serial_core.h b/include/uapi/linux/serial_core.h
|
|
index e7fe550b6..c776e947c 100644
|
|
--- a/include/uapi/linux/serial_core.h
|
|
+++ b/include/uapi/linux/serial_core.h
|
|
@@ -57,6 +57,7 @@
|
|
#define PORT_ALTR_16550_F128 28 /* Altera 16550 UART with 128 FIFOs */
|
|
#define PORT_RT2880 29 /* Ralink RT2880 internal UART */
|
|
#define PORT_16550A_FSL64 30 /* Freescale 16550 UART with 64 FIFOs */
|
|
+#define PORT_AIROHA 31 /* Airoha EN7523 internal UART */
|
|
|
|
/*
|
|
* ARM specific type numbers. These are not currently guaranteed
|
|
diff --git a/include/uapi/linux/serial_reg.h b/include/uapi/linux/serial_reg.h
|
|
index be07b5470..ebed41d12 100644
|
|
--- a/include/uapi/linux/serial_reg.h
|
|
+++ b/include/uapi/linux/serial_reg.h
|
|
@@ -376,5 +376,14 @@
|
|
#define UART_ALTR_EN_TXFIFO_LW 0x01 /* Enable the TX FIFO Low Watermark */
|
|
#define UART_ALTR_TX_LOW 0x41 /* Tx FIFO Low Watermark */
|
|
|
|
+/*
|
|
+ * These are definitions for the Airoha EN7523 uart registers
|
|
+ * Normalized because of 32 bit registers.
|
|
+ */
|
|
+#define UART_BRDL 0
|
|
+#define UART_BRDH 1
|
|
+#define UART_XINCLKDR 10
|
|
+#define UART_XYD 11
|
|
+
|
|
#endif /* _LINUX_SERIAL_REG_H */
|
|
|