mirror of
https://github.com/torvalds/linux.git
synced 2024-11-15 08:31:55 +00:00
clk: renesas: rcar-gen3: Add RPC clocks
The RPCSRC internal clock is controlled by the RPCCKCR.DIV[4:3] on all the R-Car gen3 SoCs except V3M (R8A77970) but the encoding of this field is different between SoCs; it makes sense to support the most common case of this encoding in the R-Car gen3 CPG driver... After adding the RPCSRC clock, we can add the RPC[D2] clocks derived from it and controlled by the RPCCKCR register on all the R-Car gen3 SoCs except V3M (R8A77970); the composite clock driver seems handy for this task, using the spinlock added in the previous patch... Signed-off-by: Sergei Shtylyov <sergei.shtylyov@cogentembedded.com> Signed-off-by: Geert Uytterhoeven <geert+renesas@glider.be>
This commit is contained in:
parent
875e8f6b01
commit
db4a0073cc
@ -422,6 +422,92 @@ free_clock:
|
||||
return clk;
|
||||
}
|
||||
|
||||
struct rpc_clock {
|
||||
struct clk_divider div;
|
||||
struct clk_gate gate;
|
||||
/*
|
||||
* One notifier covers both RPC and RPCD2 clocks as they are both
|
||||
* controlled by the same RPCCKCR register...
|
||||
*/
|
||||
struct cpg_simple_notifier csn;
|
||||
};
|
||||
|
||||
static const struct clk_div_table cpg_rpcsrc_div_table[] = {
|
||||
{ 2, 5 }, { 3, 6 }, { 0, 0 },
|
||||
};
|
||||
|
||||
static const struct clk_div_table cpg_rpc_div_table[] = {
|
||||
{ 1, 2 }, { 3, 4 }, { 5, 6 }, { 7, 8 }, { 0, 0 },
|
||||
};
|
||||
|
||||
static struct clk * __init cpg_rpc_clk_register(const char *name,
|
||||
void __iomem *base, const char *parent_name,
|
||||
struct raw_notifier_head *notifiers)
|
||||
{
|
||||
struct rpc_clock *rpc;
|
||||
struct clk *clk;
|
||||
|
||||
rpc = kzalloc(sizeof(*rpc), GFP_KERNEL);
|
||||
if (!rpc)
|
||||
return ERR_PTR(-ENOMEM);
|
||||
|
||||
rpc->div.reg = base + CPG_RPCCKCR;
|
||||
rpc->div.width = 3;
|
||||
rpc->div.table = cpg_rpc_div_table;
|
||||
rpc->div.lock = &cpg_lock;
|
||||
|
||||
rpc->gate.reg = base + CPG_RPCCKCR;
|
||||
rpc->gate.bit_idx = 8;
|
||||
rpc->gate.flags = CLK_GATE_SET_TO_DISABLE;
|
||||
rpc->gate.lock = &cpg_lock;
|
||||
|
||||
rpc->csn.reg = base + CPG_RPCCKCR;
|
||||
|
||||
clk = clk_register_composite(NULL, name, &parent_name, 1, NULL, NULL,
|
||||
&rpc->div.hw, &clk_divider_ops,
|
||||
&rpc->gate.hw, &clk_gate_ops, 0);
|
||||
if (IS_ERR(clk)) {
|
||||
kfree(rpc);
|
||||
return clk;
|
||||
}
|
||||
|
||||
cpg_simple_notifier_register(notifiers, &rpc->csn);
|
||||
return clk;
|
||||
}
|
||||
|
||||
struct rpcd2_clock {
|
||||
struct clk_fixed_factor fixed;
|
||||
struct clk_gate gate;
|
||||
};
|
||||
|
||||
static struct clk * __init cpg_rpcd2_clk_register(const char *name,
|
||||
void __iomem *base,
|
||||
const char *parent_name)
|
||||
{
|
||||
struct rpcd2_clock *rpcd2;
|
||||
struct clk *clk;
|
||||
|
||||
rpcd2 = kzalloc(sizeof(*rpcd2), GFP_KERNEL);
|
||||
if (!rpcd2)
|
||||
return ERR_PTR(-ENOMEM);
|
||||
|
||||
rpcd2->fixed.mult = 1;
|
||||
rpcd2->fixed.div = 2;
|
||||
|
||||
rpcd2->gate.reg = base + CPG_RPCCKCR;
|
||||
rpcd2->gate.bit_idx = 9;
|
||||
rpcd2->gate.flags = CLK_GATE_SET_TO_DISABLE;
|
||||
rpcd2->gate.lock = &cpg_lock;
|
||||
|
||||
clk = clk_register_composite(NULL, name, &parent_name, 1, NULL, NULL,
|
||||
&rpcd2->fixed.hw, &clk_fixed_factor_ops,
|
||||
&rpcd2->gate.hw, &clk_gate_ops, 0);
|
||||
if (IS_ERR(clk))
|
||||
kfree(rpcd2);
|
||||
|
||||
return clk;
|
||||
}
|
||||
|
||||
|
||||
static const struct rcar_gen3_cpg_pll_config *cpg_pll_config __initdata;
|
||||
static unsigned int cpg_clk_extalr __initdata;
|
||||
@ -600,6 +686,21 @@ struct clk * __init rcar_gen3_cpg_clk_register(struct device *dev,
|
||||
}
|
||||
break;
|
||||
|
||||
case CLK_TYPE_GEN3_RPCSRC:
|
||||
return clk_register_divider_table(NULL, core->name,
|
||||
__clk_get_name(parent), 0,
|
||||
base + CPG_RPCCKCR, 3, 2, 0,
|
||||
cpg_rpcsrc_div_table,
|
||||
&cpg_lock);
|
||||
|
||||
case CLK_TYPE_GEN3_RPC:
|
||||
return cpg_rpc_clk_register(core->name, base,
|
||||
__clk_get_name(parent), notifiers);
|
||||
|
||||
case CLK_TYPE_GEN3_RPCD2:
|
||||
return cpg_rpcd2_clk_register(core->name, base,
|
||||
__clk_get_name(parent));
|
||||
|
||||
default:
|
||||
return ERR_PTR(-EINVAL);
|
||||
}
|
||||
|
@ -23,6 +23,9 @@ enum rcar_gen3_clk_types {
|
||||
CLK_TYPE_GEN3_Z2,
|
||||
CLK_TYPE_GEN3_OSC, /* OSC EXTAL predivider and fixed divider */
|
||||
CLK_TYPE_GEN3_RCKSEL, /* Select parent/divider using RCKCR.CKSEL */
|
||||
CLK_TYPE_GEN3_RPCSRC,
|
||||
CLK_TYPE_GEN3_RPC,
|
||||
CLK_TYPE_GEN3_RPCD2,
|
||||
|
||||
/* SoC specific definitions start here */
|
||||
CLK_TYPE_GEN3_SOC_BASE,
|
||||
@ -57,6 +60,7 @@ struct rcar_gen3_cpg_pll_config {
|
||||
u8 osc_prediv;
|
||||
};
|
||||
|
||||
#define CPG_RPCCKCR 0x238
|
||||
#define CPG_RCKCR 0x240
|
||||
|
||||
struct clk *rcar_gen3_cpg_clk_register(struct device *dev,
|
||||
|
Loading…
Reference in New Issue
Block a user