mirror of
https://github.com/openwrt/openwrt.git
synced 2024-12-20 06:08:08 +00:00
8cacf2bda8
This series of upstream patches makes the system controller node as a reset provider[1][2], and it also includes some clock and reset driver fixes[3][4]. Meanwhile, all clocks and resets properties in the SoC device tree have been updated to be compatible with the new driver. [1] https://lore.kernel.org/r/20220110114930.1406665-2-sergio.paracuellos@gmail.com [2] https://lore.kernel.org/r/20220210094859.927868-2-sergio.paracuellos@gmail.com [3] https://lore.kernel.org/r/20221217074806.3225150-1-sergio.paracuellos@gmail.com [4] https://lore.kernel.org/r/20230206083305.147582-1-sergio.paracuellos@gmail.com Tested on RAISECOM MSG1500 X.00 Signed-off-by: John Thomson <git@johnthomson.fastmail.com.au> Signed-off-by: Shiji Yang <yangshiji66@qq.com>
78 lines
3.1 KiB
Diff
78 lines
3.1 KiB
Diff
From 35dcae535afc153fa83f2fe51c0812536c192c58 Mon Sep 17 00:00:00 2001
|
|
From: Sergio Paracuellos <sergio.paracuellos@gmail.com>
|
|
Date: Mon, 6 Feb 2023 09:33:05 +0100
|
|
Subject: [PATCH] clk: ralink: fix 'mt7621_gate_is_enabled()' function
|
|
|
|
Compiling clock driver with CONFIG_UBSAN enabled shows the following trace:
|
|
|
|
UBSAN: shift-out-of-bounds in drivers/clk/ralink/clk-mt7621.c:121:15
|
|
shift exponent 131072 is too large for 32-bit type 'long unsigned int'
|
|
CPU: 1 PID: 1 Comm: swapper/0 Not tainted 5.15.86 #0
|
|
Stack : ...
|
|
|
|
Call Trace:
|
|
[<80009a58>] show_stack+0x38/0x118
|
|
[<8045ce04>] dump_stack_lvl+0x60/0x80
|
|
[<80458868>] ubsan_epilogue+0x10/0x54
|
|
[<804590e0>] __ubsan_handle_shift_out_of_bounds+0x118/0x190
|
|
[<804c9a10>] mt7621_gate_is_enabled+0x98/0xa0
|
|
[<804bb774>] clk_core_is_enabled+0x34/0x90
|
|
[<80aad73c>] clk_disable_unused_subtree+0x98/0x1e4
|
|
[<80aad6d4>] clk_disable_unused_subtree+0x30/0x1e4
|
|
[<80aad6d4>] clk_disable_unused_subtree+0x30/0x1e4
|
|
[<80aad900>] clk_disable_unused+0x78/0x120
|
|
[<80002030>] do_one_initcall+0x54/0x1f0
|
|
[<80a922a4>] kernel_init_freeable+0x280/0x31c
|
|
[<808047c4>] kernel_init+0x20/0x118
|
|
[<80003e58>] ret_from_kernel_thread+0x14/0x1c
|
|
|
|
Shifting a value (131032) larger than the type (32 bit unsigned integer)
|
|
is undefined behaviour in C.
|
|
|
|
The problem is in 'mt7621_gate_is_enabled()' function which is using the
|
|
'BIT()' kernel macro with the bit index for the clock gate to check if the
|
|
bit is set. When the clock gates structure is created driver is already
|
|
setting 'bit_idx' using 'BIT()' macro, so we are wrongly applying an extra
|
|
'BIT()' mask here. Removing it solve the problem and makes this function
|
|
correct. However when clock gating is correctly working, the kernel starts
|
|
disabling those clocks that are not requested. Some drivers for this SoC
|
|
are older than this clock driver itself. So to avoid the kernel to disable
|
|
clocks that have been enabled until now, we must apply 'CLK_IS_CRITICAL'
|
|
flag on gates initialization code.
|
|
|
|
Fixes: 48df7a26f470 ("clk: ralink: add clock driver for mt7621 SoC")
|
|
Signed-off-by: Sergio Paracuellos <sergio.paracuellos@gmail.com>
|
|
Link: https://lore.kernel.org/r/20230206083305.147582-1-sergio.paracuellos@gmail.com
|
|
Signed-off-by: Stephen Boyd <sboyd@kernel.org>
|
|
---
|
|
drivers/clk/ralink/clk-mt7621.c | 10 ++++++++--
|
|
1 file changed, 8 insertions(+), 2 deletions(-)
|
|
|
|
--- a/drivers/clk/ralink/clk-mt7621.c
|
|
+++ b/drivers/clk/ralink/clk-mt7621.c
|
|
@@ -121,7 +121,7 @@ static int mt7621_gate_is_enabled(struct
|
|
if (regmap_read(sysc, SYSC_REG_CLKCFG1, &val))
|
|
return 0;
|
|
|
|
- return val & BIT(clk_gate->bit_idx);
|
|
+ return val & clk_gate->bit_idx;
|
|
}
|
|
|
|
static const struct clk_ops mt7621_gate_ops = {
|
|
@@ -133,8 +133,14 @@ static const struct clk_ops mt7621_gate_
|
|
static int mt7621_gate_ops_init(struct device *dev,
|
|
struct mt7621_gate *sclk)
|
|
{
|
|
+ /*
|
|
+ * There are drivers for this SoC that are older
|
|
+ * than clock driver and are not prepared for the clock.
|
|
+ * We don't want the kernel to disable anything so we
|
|
+ * add CLK_IS_CRITICAL flag here.
|
|
+ */
|
|
struct clk_init_data init = {
|
|
- .flags = CLK_SET_RATE_PARENT,
|
|
+ .flags = CLK_SET_RATE_PARENT | CLK_IS_CRITICAL,
|
|
.num_parents = 1,
|
|
.parent_names = &sclk->parent_name,
|
|
.ops = &mt7621_gate_ops,
|