clk: clk-gate: Create clk_gate_endisable()
This patch tries to remove duplicate code for clk_gate clocks. This creates another routine clk_gate_endisable() which will take care of enable/disable clock with knowledge of CLK_GATE_SET_TO_DISABLE flag. It works on following logic: For enabling clock, enable = 1 set2dis = 1 -> clear bit -> set = 0 set2dis = 0 -> set bit -> set = 1 For disabling clock, enable = 0 set2dis = 1 -> set bit -> set = 1 set2dis = 0 -> clear bit -> set = 0 So, result is always: enable xor set2dis. Signed-off-by: Viresh Kumar <viresh.kumar@st.com> Signed-off-by: Mike Turquette <mturquette@linaro.org>
This commit is contained in:
parent
1f73f31ad6
commit
fbc42aab54
1 changed files with 25 additions and 29 deletions
|
@ -28,32 +28,38 @@
|
||||||
|
|
||||||
#define to_clk_gate(_hw) container_of(_hw, struct clk_gate, hw)
|
#define to_clk_gate(_hw) container_of(_hw, struct clk_gate, hw)
|
||||||
|
|
||||||
static void clk_gate_set_bit(struct clk_gate *gate)
|
/*
|
||||||
|
* It works on following logic:
|
||||||
|
*
|
||||||
|
* For enabling clock, enable = 1
|
||||||
|
* set2dis = 1 -> clear bit -> set = 0
|
||||||
|
* set2dis = 0 -> set bit -> set = 1
|
||||||
|
*
|
||||||
|
* For disabling clock, enable = 0
|
||||||
|
* set2dis = 1 -> set bit -> set = 1
|
||||||
|
* set2dis = 0 -> clear bit -> set = 0
|
||||||
|
*
|
||||||
|
* So, result is always: enable xor set2dis.
|
||||||
|
*/
|
||||||
|
static void clk_gate_endisable(struct clk_hw *hw, int enable)
|
||||||
{
|
{
|
||||||
u32 reg;
|
struct clk_gate *gate = to_clk_gate(hw);
|
||||||
|
int set = gate->flags & CLK_GATE_SET_TO_DISABLE ? 1 : 0;
|
||||||
unsigned long flags = 0;
|
unsigned long flags = 0;
|
||||||
|
u32 reg;
|
||||||
|
|
||||||
|
set ^= enable;
|
||||||
|
|
||||||
if (gate->lock)
|
if (gate->lock)
|
||||||
spin_lock_irqsave(gate->lock, flags);
|
spin_lock_irqsave(gate->lock, flags);
|
||||||
|
|
||||||
reg = readl(gate->reg);
|
reg = readl(gate->reg);
|
||||||
reg |= BIT(gate->bit_idx);
|
|
||||||
writel(reg, gate->reg);
|
|
||||||
|
|
||||||
if (gate->lock)
|
if (set)
|
||||||
spin_unlock_irqrestore(gate->lock, flags);
|
reg |= BIT(gate->bit_idx);
|
||||||
}
|
else
|
||||||
|
reg &= ~BIT(gate->bit_idx);
|
||||||
|
|
||||||
static void clk_gate_clear_bit(struct clk_gate *gate)
|
|
||||||
{
|
|
||||||
u32 reg;
|
|
||||||
unsigned long flags = 0;
|
|
||||||
|
|
||||||
if (gate->lock)
|
|
||||||
spin_lock_irqsave(gate->lock, flags);
|
|
||||||
|
|
||||||
reg = readl(gate->reg);
|
|
||||||
reg &= ~BIT(gate->bit_idx);
|
|
||||||
writel(reg, gate->reg);
|
writel(reg, gate->reg);
|
||||||
|
|
||||||
if (gate->lock)
|
if (gate->lock)
|
||||||
|
@ -62,24 +68,14 @@ static void clk_gate_clear_bit(struct clk_gate *gate)
|
||||||
|
|
||||||
static int clk_gate_enable(struct clk_hw *hw)
|
static int clk_gate_enable(struct clk_hw *hw)
|
||||||
{
|
{
|
||||||
struct clk_gate *gate = to_clk_gate(hw);
|
clk_gate_endisable(hw, 1);
|
||||||
|
|
||||||
if (gate->flags & CLK_GATE_SET_TO_DISABLE)
|
|
||||||
clk_gate_clear_bit(gate);
|
|
||||||
else
|
|
||||||
clk_gate_set_bit(gate);
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void clk_gate_disable(struct clk_hw *hw)
|
static void clk_gate_disable(struct clk_hw *hw)
|
||||||
{
|
{
|
||||||
struct clk_gate *gate = to_clk_gate(hw);
|
clk_gate_endisable(hw, 0);
|
||||||
|
|
||||||
if (gate->flags & CLK_GATE_SET_TO_DISABLE)
|
|
||||||
clk_gate_set_bit(gate);
|
|
||||||
else
|
|
||||||
clk_gate_clear_bit(gate);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int clk_gate_is_enabled(struct clk_hw *hw)
|
static int clk_gate_is_enabled(struct clk_hw *hw)
|
||||||
|
|
Loading…
Add table
Reference in a new issue