Commit ec240f88 authored by Valentin Caron's avatar Valentin Caron Committed by Greg Kroah-Hartman

serial: stm32: implement prescaler tuning to compute low baudrate

In the case of high USART input clock and low baud rate, BRR value
is not enough to get correct baud rate. So here we use USART prescaler to
divide USART input clock to get the correct baud rate.

PRESC register is only available since stm32h7.
Signed-off-by: default avatarValentin Caron <valentin.caron@foss.st.com>
Link: https://lore.kernel.org/r/20240112095300.2004878-2-valentin.caron@foss.st.comSigned-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 5c49b6a4
...@@ -9,6 +9,7 @@ ...@@ -9,6 +9,7 @@
* Inspired by st-asc.c from STMicroelectronics (c) * Inspired by st-asc.c from STMicroelectronics (c)
*/ */
#include <linux/bitfield.h>
#include <linux/clk.h> #include <linux/clk.h>
#include <linux/console.h> #include <linux/console.h>
#include <linux/delay.h> #include <linux/delay.h>
...@@ -50,6 +51,7 @@ static struct stm32_usart_info __maybe_unused stm32f4_info = { ...@@ -50,6 +51,7 @@ static struct stm32_usart_info __maybe_unused stm32f4_info = {
.rtor = UNDEF_REG, .rtor = UNDEF_REG,
.rqr = UNDEF_REG, .rqr = UNDEF_REG,
.icr = UNDEF_REG, .icr = UNDEF_REG,
.presc = UNDEF_REG,
}, },
.cfg = { .cfg = {
.uart_enable_bit = 13, .uart_enable_bit = 13,
...@@ -71,6 +73,7 @@ static struct stm32_usart_info __maybe_unused stm32f7_info = { ...@@ -71,6 +73,7 @@ static struct stm32_usart_info __maybe_unused stm32f7_info = {
.icr = 0x20, .icr = 0x20,
.rdr = 0x24, .rdr = 0x24,
.tdr = 0x28, .tdr = 0x28,
.presc = UNDEF_REG,
}, },
.cfg = { .cfg = {
.uart_enable_bit = 0, .uart_enable_bit = 0,
...@@ -93,6 +96,7 @@ static struct stm32_usart_info __maybe_unused stm32h7_info = { ...@@ -93,6 +96,7 @@ static struct stm32_usart_info __maybe_unused stm32h7_info = {
.icr = 0x20, .icr = 0x20,
.rdr = 0x24, .rdr = 0x24,
.tdr = 0x28, .tdr = 0x28,
.presc = 0x2c,
}, },
.cfg = { .cfg = {
.uart_enable_bit = 0, .uart_enable_bit = 0,
...@@ -1145,6 +1149,8 @@ static void stm32_usart_shutdown(struct uart_port *port) ...@@ -1145,6 +1149,8 @@ static void stm32_usart_shutdown(struct uart_port *port)
free_irq(port->irq, port); free_irq(port->irq, port);
} }
static const unsigned int stm32_usart_presc_val[] = {1, 2, 4, 6, 8, 10, 12, 16, 32, 64, 128, 256};
static void stm32_usart_set_termios(struct uart_port *port, static void stm32_usart_set_termios(struct uart_port *port,
struct ktermios *termios, struct ktermios *termios,
const struct ktermios *old) const struct ktermios *old)
...@@ -1153,17 +1159,19 @@ static void stm32_usart_set_termios(struct uart_port *port, ...@@ -1153,17 +1159,19 @@ static void stm32_usart_set_termios(struct uart_port *port,
const struct stm32_usart_offsets *ofs = &stm32_port->info->ofs; const struct stm32_usart_offsets *ofs = &stm32_port->info->ofs;
const struct stm32_usart_config *cfg = &stm32_port->info->cfg; const struct stm32_usart_config *cfg = &stm32_port->info->cfg;
struct serial_rs485 *rs485conf = &port->rs485; struct serial_rs485 *rs485conf = &port->rs485;
unsigned int baud, bits; unsigned int baud, bits, uart_clk, uart_clk_pres;
u32 usartdiv, mantissa, fraction, oversampling; u32 usartdiv, mantissa, fraction, oversampling;
tcflag_t cflag = termios->c_cflag; tcflag_t cflag = termios->c_cflag;
u32 cr1, cr2, cr3, isr; u32 cr1, cr2, cr3, isr, brr, presc;
unsigned long flags; unsigned long flags;
int ret; int ret;
if (!stm32_port->hw_flow_control) if (!stm32_port->hw_flow_control)
cflag &= ~CRTSCTS; cflag &= ~CRTSCTS;
baud = uart_get_baud_rate(port, termios, old, 0, port->uartclk / 8); uart_clk = clk_get_rate(stm32_port->clk);
baud = uart_get_baud_rate(port, termios, old, 0, uart_clk / 8);
uart_port_lock_irqsave(port, &flags); uart_port_lock_irqsave(port, &flags);
...@@ -1265,27 +1273,48 @@ static void stm32_usart_set_termios(struct uart_port *port, ...@@ -1265,27 +1273,48 @@ static void stm32_usart_set_termios(struct uart_port *port,
cr3 |= USART_CR3_CTSE | USART_CR3_RTSE; cr3 |= USART_CR3_CTSE | USART_CR3_RTSE;
} }
usartdiv = DIV_ROUND_CLOSEST(port->uartclk, baud); for (presc = 0; presc <= USART_PRESC_MAX; presc++) {
uart_clk_pres = DIV_ROUND_CLOSEST(uart_clk, stm32_usart_presc_val[presc]);
usartdiv = DIV_ROUND_CLOSEST(uart_clk_pres, baud);
/* /*
* The USART supports 16 or 8 times oversampling. * The USART supports 16 or 8 times oversampling.
* By default we prefer 16 times oversampling, so that the receiver * By default we prefer 16 times oversampling, so that the receiver
* has a better tolerance to clock deviations. * has a better tolerance to clock deviations.
* 8 times oversampling is only used to achieve higher speeds. * 8 times oversampling is only used to achieve higher speeds.
*/ */
if (usartdiv < 16) { if (usartdiv < 16) {
oversampling = 8; oversampling = 8;
cr1 |= USART_CR1_OVER8; cr1 |= USART_CR1_OVER8;
stm32_usart_set_bits(port, ofs->cr1, USART_CR1_OVER8); stm32_usart_set_bits(port, ofs->cr1, USART_CR1_OVER8);
} else { } else {
oversampling = 16; oversampling = 16;
cr1 &= ~USART_CR1_OVER8; cr1 &= ~USART_CR1_OVER8;
stm32_usart_clr_bits(port, ofs->cr1, USART_CR1_OVER8); stm32_usart_clr_bits(port, ofs->cr1, USART_CR1_OVER8);
}
mantissa = (usartdiv / oversampling) << USART_BRR_DIV_M_SHIFT;
fraction = usartdiv % oversampling;
brr = mantissa | fraction;
if (FIELD_FIT(USART_BRR_MASK, brr)) {
if (ofs->presc != UNDEF_REG) {
port->uartclk = uart_clk_pres;
writel_relaxed(presc, port->membase + ofs->presc);
} else if (presc) {
/* We need a prescaler but we don't have it (STM32F4, STM32F7) */
dev_err(port->dev,
"unable to set baudrate, input clock is too high");
}
break;
} else if (presc == USART_PRESC_MAX) {
/* Even with prescaler and brr at max value we can't set baudrate */
dev_err(port->dev, "unable to set baudrate, input clock is too high");
break;
}
} }
mantissa = (usartdiv / oversampling) << USART_BRR_DIV_M_SHIFT; writel_relaxed(brr, port->membase + ofs->brr);
fraction = usartdiv % oversampling;
writel_relaxed(mantissa | fraction, port->membase + ofs->brr);
uart_update_timeout(port, cflag, baud); uart_update_timeout(port, cflag, baud);
......
...@@ -20,6 +20,7 @@ struct stm32_usart_offsets { ...@@ -20,6 +20,7 @@ struct stm32_usart_offsets {
u8 icr; u8 icr;
u8 rdr; u8 rdr;
u8 tdr; u8 tdr;
u8 presc;
}; };
struct stm32_usart_config { struct stm32_usart_config {
...@@ -71,6 +72,7 @@ struct stm32_usart_info { ...@@ -71,6 +72,7 @@ struct stm32_usart_info {
#define USART_BRR_DIV_M_MASK GENMASK(15, 4) #define USART_BRR_DIV_M_MASK GENMASK(15, 4)
#define USART_BRR_DIV_M_SHIFT 4 #define USART_BRR_DIV_M_SHIFT 4
#define USART_BRR_04_R_SHIFT 1 #define USART_BRR_04_R_SHIFT 1
#define USART_BRR_MASK (USART_BRR_DIV_M_MASK | USART_BRR_DIV_F_MASK)
/* USART_CR1 */ /* USART_CR1 */
#define USART_CR1_SBK BIT(0) #define USART_CR1_SBK BIT(0)
...@@ -176,6 +178,10 @@ struct stm32_usart_info { ...@@ -176,6 +178,10 @@ struct stm32_usart_info {
#define USART_ICR_CMCF BIT(17) /* F7 */ #define USART_ICR_CMCF BIT(17) /* F7 */
#define USART_ICR_WUCF BIT(20) /* H7 */ #define USART_ICR_WUCF BIT(20) /* H7 */
/* USART_PRESC */
#define USART_PRESC GENMASK(3, 0) /* H7 */
#define USART_PRESC_MAX 0b1011
#define STM32_SERIAL_NAME "ttySTM" #define STM32_SERIAL_NAME "ttySTM"
#define STM32_MAX_PORTS 8 #define STM32_MAX_PORTS 8
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment