mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-12-17 16:14:25 +08:00
serial: imx: Ensure that imx_uart_rs485_config() is called with enabled clock
There are register accesses in the function imx_uart_rs485_config(). The clock must be enabled for these accesses. This was ensured by calling it via the function uart_rs485_config() in the probe() function within the range where the clock is enabled. With the commit7c7f9bc986
("serial: Deassert Transmit Enable on probe in driver-specific way") it was removed from the probe() function and is now only called through the function uart_add_one_port() which is located at the end of the probe() function. But the clock is already switched off in this area. To ensure that the clock is enabled during register access, move the disabling of the clock to the very end of the probe() function. To avoid leaking enabled clocks on error also add an error path for exiting with disabling the clock. Fixes:7c7f9bc986
("serial: Deassert Transmit Enable on probe in driver-specific way") Cc: stable <stable@kernel.org> Signed-off-by: Christoph Niedermaier <cniedermaier@dh-electronics.com> Reviewed-by: Lukas Wunner <lukas@wunner.de> Link: https://lore.kernel.org/r/20231226113647.39376-1-cniedermaier@dh-electronics.com Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
c6dcd8050f
commit
7c45eaa813
@ -2332,10 +2332,8 @@ static int imx_uart_probe(struct platform_device *pdev)
|
||||
}
|
||||
|
||||
ret = uart_get_rs485_mode(&sport->port);
|
||||
if (ret) {
|
||||
clk_disable_unprepare(sport->clk_ipg);
|
||||
return ret;
|
||||
}
|
||||
if (ret)
|
||||
goto err_clk;
|
||||
|
||||
if (sport->port.rs485.flags & SER_RS485_ENABLED &&
|
||||
(!sport->have_rtscts && !sport->have_rtsgpio))
|
||||
@ -2419,8 +2417,6 @@ static int imx_uart_probe(struct platform_device *pdev)
|
||||
imx_uart_writel(sport, ucr3, UCR3);
|
||||
}
|
||||
|
||||
clk_disable_unprepare(sport->clk_ipg);
|
||||
|
||||
hrtimer_init(&sport->trigger_start_tx, CLOCK_MONOTONIC, HRTIMER_MODE_REL);
|
||||
hrtimer_init(&sport->trigger_stop_tx, CLOCK_MONOTONIC, HRTIMER_MODE_REL);
|
||||
sport->trigger_start_tx.function = imx_trigger_start_tx;
|
||||
@ -2436,7 +2432,7 @@ static int imx_uart_probe(struct platform_device *pdev)
|
||||
if (ret) {
|
||||
dev_err(&pdev->dev, "failed to request rx irq: %d\n",
|
||||
ret);
|
||||
return ret;
|
||||
goto err_clk;
|
||||
}
|
||||
|
||||
ret = devm_request_irq(&pdev->dev, txirq, imx_uart_txint, 0,
|
||||
@ -2444,7 +2440,7 @@ static int imx_uart_probe(struct platform_device *pdev)
|
||||
if (ret) {
|
||||
dev_err(&pdev->dev, "failed to request tx irq: %d\n",
|
||||
ret);
|
||||
return ret;
|
||||
goto err_clk;
|
||||
}
|
||||
|
||||
ret = devm_request_irq(&pdev->dev, rtsirq, imx_uart_rtsint, 0,
|
||||
@ -2452,14 +2448,14 @@ static int imx_uart_probe(struct platform_device *pdev)
|
||||
if (ret) {
|
||||
dev_err(&pdev->dev, "failed to request rts irq: %d\n",
|
||||
ret);
|
||||
return ret;
|
||||
goto err_clk;
|
||||
}
|
||||
} else {
|
||||
ret = devm_request_irq(&pdev->dev, rxirq, imx_uart_int, 0,
|
||||
dev_name(&pdev->dev), sport);
|
||||
if (ret) {
|
||||
dev_err(&pdev->dev, "failed to request irq: %d\n", ret);
|
||||
return ret;
|
||||
goto err_clk;
|
||||
}
|
||||
}
|
||||
|
||||
@ -2467,7 +2463,12 @@ static int imx_uart_probe(struct platform_device *pdev)
|
||||
|
||||
platform_set_drvdata(pdev, sport);
|
||||
|
||||
return uart_add_one_port(&imx_uart_uart_driver, &sport->port);
|
||||
ret = uart_add_one_port(&imx_uart_uart_driver, &sport->port);
|
||||
|
||||
err_clk:
|
||||
clk_disable_unprepare(sport->clk_ipg);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void imx_uart_remove(struct platform_device *pdev)
|
||||
|
Loading…
Reference in New Issue
Block a user