|
@@ -160,13 +160,81 @@ static unsigned long sh_clk_div4_recalc(struct clk *clk)
|
|
|
return clk->freq_table[idx].frequency;
|
|
|
}
|
|
|
|
|
|
+static int sh_clk_div4_set_parent(struct clk *clk, struct clk *parent)
|
|
|
+{
|
|
|
+ struct clk_div_mult_table *table = clk->priv;
|
|
|
+ u32 value;
|
|
|
+ int ret;
|
|
|
+
|
|
|
+ if (!strcmp("pll_clk", parent->name))
|
|
|
+ value = __raw_readl(clk->enable_reg) & ~(1 << 7);
|
|
|
+ else
|
|
|
+ value = __raw_readl(clk->enable_reg) | (1 << 7);
|
|
|
+
|
|
|
+ ret = clk_reparent(clk, parent);
|
|
|
+ if (ret < 0)
|
|
|
+ return ret;
|
|
|
+
|
|
|
+ __raw_writel(value, clk->enable_reg);
|
|
|
+
|
|
|
+ /* Rebiuld the frequency table */
|
|
|
+ clk_rate_table_build(clk, clk->freq_table, table->nr_divisors,
|
|
|
+ table, &clk->arch_flags);
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+static int sh_clk_div4_set_rate(struct clk *clk, unsigned long rate, int algo_id)
|
|
|
+{
|
|
|
+ unsigned long value;
|
|
|
+ int idx = clk_rate_table_find(clk, clk->freq_table, rate);
|
|
|
+ if (idx < 0)
|
|
|
+ return idx;
|
|
|
+
|
|
|
+ value = __raw_readl(clk->enable_reg);
|
|
|
+ value &= ~0xf;
|
|
|
+ value |= idx;
|
|
|
+ __raw_writel(value, clk->enable_reg);
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+static int sh_clk_div4_enable(struct clk *clk)
|
|
|
+{
|
|
|
+ __raw_writel(__raw_readl(clk->enable_reg) & ~(1 << 8), clk->enable_reg);
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+static void sh_clk_div4_disable(struct clk *clk)
|
|
|
+{
|
|
|
+ __raw_writel(__raw_readl(clk->enable_reg) | (1 << 8), clk->enable_reg);
|
|
|
+}
|
|
|
+
|
|
|
static struct clk_ops sh_clk_div4_clk_ops = {
|
|
|
.recalc = sh_clk_div4_recalc,
|
|
|
+ .set_rate = sh_clk_div4_set_rate,
|
|
|
.round_rate = sh_clk_div_round_rate,
|
|
|
};
|
|
|
|
|
|
-int __init sh_clk_div4_register(struct clk *clks, int nr,
|
|
|
- struct clk_div_mult_table *table)
|
|
|
+static struct clk_ops sh_clk_div4_enable_clk_ops = {
|
|
|
+ .recalc = sh_clk_div4_recalc,
|
|
|
+ .set_rate = sh_clk_div4_set_rate,
|
|
|
+ .round_rate = sh_clk_div_round_rate,
|
|
|
+ .enable = sh_clk_div4_enable,
|
|
|
+ .disable = sh_clk_div4_disable,
|
|
|
+};
|
|
|
+
|
|
|
+static struct clk_ops sh_clk_div4_reparent_clk_ops = {
|
|
|
+ .recalc = sh_clk_div4_recalc,
|
|
|
+ .set_rate = sh_clk_div4_set_rate,
|
|
|
+ .round_rate = sh_clk_div_round_rate,
|
|
|
+ .enable = sh_clk_div4_enable,
|
|
|
+ .disable = sh_clk_div4_disable,
|
|
|
+ .set_parent = sh_clk_div4_set_parent,
|
|
|
+};
|
|
|
+
|
|
|
+static int __init sh_clk_div4_register_ops(struct clk *clks, int nr,
|
|
|
+ struct clk_div_mult_table *table, struct clk_ops *ops)
|
|
|
{
|
|
|
struct clk *clkp;
|
|
|
void *freq_table;
|
|
@@ -185,7 +253,7 @@ int __init sh_clk_div4_register(struct clk *clks, int nr,
|
|
|
for (k = 0; !ret && (k < nr); k++) {
|
|
|
clkp = clks + k;
|
|
|
|
|
|
- clkp->ops = &sh_clk_div4_clk_ops;
|
|
|
+ clkp->ops = ops;
|
|
|
clkp->id = -1;
|
|
|
clkp->priv = table;
|
|
|
|
|
@@ -198,6 +266,26 @@ int __init sh_clk_div4_register(struct clk *clks, int nr,
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
+int __init sh_clk_div4_register(struct clk *clks, int nr,
|
|
|
+ struct clk_div_mult_table *table)
|
|
|
+{
|
|
|
+ return sh_clk_div4_register_ops(clks, nr, table, &sh_clk_div4_clk_ops);
|
|
|
+}
|
|
|
+
|
|
|
+int __init sh_clk_div4_enable_register(struct clk *clks, int nr,
|
|
|
+ struct clk_div_mult_table *table)
|
|
|
+{
|
|
|
+ return sh_clk_div4_register_ops(clks, nr, table,
|
|
|
+ &sh_clk_div4_enable_clk_ops);
|
|
|
+}
|
|
|
+
|
|
|
+int __init sh_clk_div4_reparent_register(struct clk *clks, int nr,
|
|
|
+ struct clk_div_mult_table *table)
|
|
|
+{
|
|
|
+ return sh_clk_div4_register_ops(clks, nr, table,
|
|
|
+ &sh_clk_div4_reparent_clk_ops);
|
|
|
+}
|
|
|
+
|
|
|
#ifdef CONFIG_SH_CLK_CPG_LEGACY
|
|
|
static struct clk master_clk = {
|
|
|
.name = "master_clk",
|