mirror of
https://github.com/AuxXxilium/linux_dsm_epyc7002.git
synced 2025-04-04 11:48:14 +07:00
IMX UART: do not assume 16MHz reference frequency
We assumed a 16MHz reference frequency for the UART. While this is true for i.MX1 most of the time it is not true for MX27/MX31. Also, add handling for the ONEMS register which is present on newer versions of the chip and pass a sane minimum baudrate to uart_get_baud_rate(). Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de>
This commit is contained in:
parent
63dd10846d
commit
036bb15ec9
@ -589,6 +589,7 @@ imx_set_termios(struct uart_port *port, struct ktermios *termios,
|
|||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
unsigned int ucr2, old_ucr1, old_txrxen, baud, quot;
|
unsigned int ucr2, old_ucr1, old_txrxen, baud, quot;
|
||||||
unsigned int old_csize = old ? old->c_cflag & CSIZE : CS8;
|
unsigned int old_csize = old ? old->c_cflag & CSIZE : CS8;
|
||||||
|
unsigned int div, num, denom, ufcr;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If we don't support modem control lines, don't allow
|
* If we don't support modem control lines, don't allow
|
||||||
@ -634,7 +635,7 @@ imx_set_termios(struct uart_port *port, struct ktermios *termios,
|
|||||||
/*
|
/*
|
||||||
* Ask the core to calculate the divisor for us.
|
* Ask the core to calculate the divisor for us.
|
||||||
*/
|
*/
|
||||||
baud = uart_get_baud_rate(port, termios, old, 0, port->uartclk/16);
|
baud = uart_get_baud_rate(port, termios, old, 50, port->uartclk / 16);
|
||||||
quot = uart_get_divisor(port, baud);
|
quot = uart_get_divisor(port, baud);
|
||||||
|
|
||||||
spin_lock_irqsave(&sport->port.lock, flags);
|
spin_lock_irqsave(&sport->port.lock, flags);
|
||||||
@ -684,14 +685,41 @@ imx_set_termios(struct uart_port *port, struct ktermios *termios,
|
|||||||
sport->port.membase + UCR2);
|
sport->port.membase + UCR2);
|
||||||
old_txrxen &= (UCR2_TXEN | UCR2_RXEN);
|
old_txrxen &= (UCR2_TXEN | UCR2_RXEN);
|
||||||
|
|
||||||
/* set the baud rate. We assume uartclk = 16 MHz
|
div = sport->port.uartclk / (baud * 16);
|
||||||
*
|
if (div > 7)
|
||||||
* baud * 16 UBIR - 1
|
div = 7;
|
||||||
* --------- = --------
|
if (!div)
|
||||||
* uartclk UBMR - 1
|
div = 1;
|
||||||
*/
|
|
||||||
writel((baud / 100) - 1, sport->port.membase + UBIR);
|
num = baud;
|
||||||
writel(10000 - 1, sport->port.membase + UBMR);
|
denom = port->uartclk / div / 16;
|
||||||
|
|
||||||
|
/* shift num and denom right until they fit into 16 bits */
|
||||||
|
while (num > 0x10000 || denom > 0x10000) {
|
||||||
|
num >>= 1;
|
||||||
|
denom >>= 1;
|
||||||
|
}
|
||||||
|
if (num > 0)
|
||||||
|
num -= 1;
|
||||||
|
if (denom > 0)
|
||||||
|
denom -= 1;
|
||||||
|
|
||||||
|
writel(num, sport->port.membase + UBIR);
|
||||||
|
writel(denom, sport->port.membase + UBMR);
|
||||||
|
|
||||||
|
if (div == 7)
|
||||||
|
div = 6; /* 6 in RFDIV means divide by 7 */
|
||||||
|
else
|
||||||
|
div = 6 - div;
|
||||||
|
|
||||||
|
ufcr = readl(sport->port.membase + UFCR);
|
||||||
|
ufcr = (ufcr & (~UFCR_RFDIV)) |
|
||||||
|
(div << 7);
|
||||||
|
writel(ufcr, sport->port.membase + UFCR);
|
||||||
|
|
||||||
|
#ifdef ONEMS
|
||||||
|
writel(sport->port.uartclk / div / 1000, sport->port.membase + ONEMS);
|
||||||
|
#endif
|
||||||
|
|
||||||
writel(old_ucr1, sport->port.membase + UCR1);
|
writel(old_ucr1, sport->port.membase + UCR1);
|
||||||
|
|
||||||
@ -812,7 +840,6 @@ static struct imx_port imx_ports[] = {
|
|||||||
.membase = (void *)IMX_UART1_BASE,
|
.membase = (void *)IMX_UART1_BASE,
|
||||||
.mapbase = 0x00206000,
|
.mapbase = 0x00206000,
|
||||||
.irq = UART1_MINT_RX,
|
.irq = UART1_MINT_RX,
|
||||||
.uartclk = 16000000,
|
|
||||||
.fifosize = 32,
|
.fifosize = 32,
|
||||||
.flags = UPF_BOOT_AUTOCONF,
|
.flags = UPF_BOOT_AUTOCONF,
|
||||||
.ops = &imx_pops,
|
.ops = &imx_pops,
|
||||||
@ -828,7 +855,6 @@ static struct imx_port imx_ports[] = {
|
|||||||
.membase = (void *)IMX_UART2_BASE,
|
.membase = (void *)IMX_UART2_BASE,
|
||||||
.mapbase = 0x00207000,
|
.mapbase = 0x00207000,
|
||||||
.irq = UART2_MINT_RX,
|
.irq = UART2_MINT_RX,
|
||||||
.uartclk = 16000000,
|
|
||||||
.fifosize = 32,
|
.fifosize = 32,
|
||||||
.flags = UPF_BOOT_AUTOCONF,
|
.flags = UPF_BOOT_AUTOCONF,
|
||||||
.ops = &imx_pops,
|
.ops = &imx_pops,
|
||||||
@ -858,6 +884,8 @@ static void __init imx_init_ports(void)
|
|||||||
init_timer(&imx_ports[i].timer);
|
init_timer(&imx_ports[i].timer);
|
||||||
imx_ports[i].timer.function = imx_timeout;
|
imx_ports[i].timer.function = imx_timeout;
|
||||||
imx_ports[i].timer.data = (unsigned long)&imx_ports[i];
|
imx_ports[i].timer.data = (unsigned long)&imx_ports[i];
|
||||||
|
|
||||||
|
imx_ports[i].port.uartclk = imx_get_perclk1();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user