mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-11-16 08:44:21 +08:00
clk: shmobile: div6: Avoid changing divisor in .disable()
While DIV6 clocks require the divisor field to be non-zero when stopping the clock, some clocks (e.g. ZB on sh73a0) fail to be re-enabled later if the divisor field is changed when stopping the clock. The reason for this is unknown. To fix this, do not touch the divisor field if it's already non-zero. On kzm9g, the smsc911x Ethernet controller is connected to the sh73a0 Bus State Controller, which is clocked by the ZB clock. Without this fix, if the ZB clock is disabled during system suspend, and re-enabled during resume, the kernel locks up when the smsc911x driver tries to access the Ethernet registers. Signed-off-by: Geert Uytterhoeven <geert+renesas@glider.be> Acked-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
This commit is contained in:
parent
97bf6af1f9
commit
7980a86190
@ -54,12 +54,19 @@ static int cpg_div6_clock_enable(struct clk_hw *hw)
|
||||
static void cpg_div6_clock_disable(struct clk_hw *hw)
|
||||
{
|
||||
struct div6_clock *clock = to_div6_clock(hw);
|
||||
u32 val;
|
||||
|
||||
/* DIV6 clocks require the divisor field to be non-zero when stopping
|
||||
* the clock.
|
||||
val = clk_readl(clock->reg);
|
||||
val |= CPG_DIV6_CKSTP;
|
||||
/*
|
||||
* DIV6 clocks require the divisor field to be non-zero when stopping
|
||||
* the clock. However, some clocks (e.g. ZB on sh73a0) fail to be
|
||||
* re-enabled later if the divisor field is changed when stopping the
|
||||
* clock
|
||||
*/
|
||||
clk_writel(clk_readl(clock->reg) | CPG_DIV6_CKSTP | CPG_DIV6_DIV_MASK,
|
||||
clock->reg);
|
||||
if (!(val & CPG_DIV6_DIV_MASK))
|
||||
val |= CPG_DIV6_DIV_MASK;
|
||||
clk_writel(val, clock->reg);
|
||||
}
|
||||
|
||||
static int cpg_div6_clock_is_enabled(struct clk_hw *hw)
|
||||
|
Loading…
Reference in New Issue
Block a user