Commit c642e6a9 authored by Thomas Petazzoni's avatar Thomas Petazzoni Committed by Jason Cooper

clk: mvebu: refactor corediv driver to support more SoC

This commit refactors the corediv clock driver so that it is capable
of handling various SOCs that have slightly different corediv clock
registers and capabilities.

It introduces a clk_corediv_soc_desc structure that encapsulates all
the SoC specific details.
Signed-off-by: default avatarThomas Petazzoni <thomas.petazzoni@free-electrons.com>
Signed-off-by: default avatarEzequiel Garcia <ezequiel.garcia@free-electrons.com>
Signed-off-by: default avatarJason Cooper <jason@lakedaemon.net>
parent 846f33e6
...@@ -18,9 +18,6 @@ ...@@ -18,9 +18,6 @@
#include "common.h" #include "common.h"
#define CORE_CLK_DIV_RATIO_MASK 0xff #define CORE_CLK_DIV_RATIO_MASK 0xff
#define CORE_CLK_DIV_RATIO_RELOAD BIT(8)
#define CORE_CLK_DIV_ENABLE_OFFSET 24
#define CORE_CLK_DIV_RATIO_OFFSET 0x8
/* /*
* This structure describes the hardware details (bit offset and mask) * This structure describes the hardware details (bit offset and mask)
...@@ -35,6 +32,21 @@ struct clk_corediv_desc { ...@@ -35,6 +32,21 @@ struct clk_corediv_desc {
unsigned int fieldbit; unsigned int fieldbit;
}; };
/*
* This structure describes the hardware details to configure the core
* divider clocks on a given SoC. Amongst others, it points to the
* array of core divider clock descriptors for this SoC, as well as
* the corresponding operations to manipulate them.
*/
struct clk_corediv_soc_desc {
const struct clk_corediv_desc *descs;
unsigned int ndescs;
const struct clk_ops ops;
u32 ratio_reload;
u32 enable_bit_offset;
u32 ratio_offset;
};
/* /*
* This structure represents one core divider clock for the clock * This structure represents one core divider clock for the clock
* framework, and is dynamically allocated for each core divider clock * framework, and is dynamically allocated for each core divider clock
...@@ -44,6 +56,7 @@ struct clk_corediv { ...@@ -44,6 +56,7 @@ struct clk_corediv {
struct clk_hw hw; struct clk_hw hw;
void __iomem *reg; void __iomem *reg;
const struct clk_corediv_desc *desc; const struct clk_corediv_desc *desc;
const struct clk_corediv_soc_desc *soc_desc;
spinlock_t lock; spinlock_t lock;
}; };
...@@ -63,8 +76,9 @@ static const struct clk_corediv_desc mvebu_corediv_desc[] = { ...@@ -63,8 +76,9 @@ static const struct clk_corediv_desc mvebu_corediv_desc[] = {
static int clk_corediv_is_enabled(struct clk_hw *hwclk) static int clk_corediv_is_enabled(struct clk_hw *hwclk)
{ {
struct clk_corediv *corediv = to_corediv_clk(hwclk); struct clk_corediv *corediv = to_corediv_clk(hwclk);
const struct clk_corediv_soc_desc *soc_desc = corediv->soc_desc;
const struct clk_corediv_desc *desc = corediv->desc; const struct clk_corediv_desc *desc = corediv->desc;
u32 enable_mask = BIT(desc->fieldbit) << CORE_CLK_DIV_ENABLE_OFFSET; u32 enable_mask = BIT(desc->fieldbit) << soc_desc->enable_bit_offset;
return !!(readl(corediv->reg) & enable_mask); return !!(readl(corediv->reg) & enable_mask);
} }
...@@ -72,6 +86,7 @@ static int clk_corediv_is_enabled(struct clk_hw *hwclk) ...@@ -72,6 +86,7 @@ static int clk_corediv_is_enabled(struct clk_hw *hwclk)
static int clk_corediv_enable(struct clk_hw *hwclk) static int clk_corediv_enable(struct clk_hw *hwclk)
{ {
struct clk_corediv *corediv = to_corediv_clk(hwclk); struct clk_corediv *corediv = to_corediv_clk(hwclk);
const struct clk_corediv_soc_desc *soc_desc = corediv->soc_desc;
const struct clk_corediv_desc *desc = corediv->desc; const struct clk_corediv_desc *desc = corediv->desc;
unsigned long flags = 0; unsigned long flags = 0;
u32 reg; u32 reg;
...@@ -79,7 +94,7 @@ static int clk_corediv_enable(struct clk_hw *hwclk) ...@@ -79,7 +94,7 @@ static int clk_corediv_enable(struct clk_hw *hwclk)
spin_lock_irqsave(&corediv->lock, flags); spin_lock_irqsave(&corediv->lock, flags);
reg = readl(corediv->reg); reg = readl(corediv->reg);
reg |= (BIT(desc->fieldbit) << CORE_CLK_DIV_ENABLE_OFFSET); reg |= (BIT(desc->fieldbit) << soc_desc->enable_bit_offset);
writel(reg, corediv->reg); writel(reg, corediv->reg);
spin_unlock_irqrestore(&corediv->lock, flags); spin_unlock_irqrestore(&corediv->lock, flags);
...@@ -90,6 +105,7 @@ static int clk_corediv_enable(struct clk_hw *hwclk) ...@@ -90,6 +105,7 @@ static int clk_corediv_enable(struct clk_hw *hwclk)
static void clk_corediv_disable(struct clk_hw *hwclk) static void clk_corediv_disable(struct clk_hw *hwclk)
{ {
struct clk_corediv *corediv = to_corediv_clk(hwclk); struct clk_corediv *corediv = to_corediv_clk(hwclk);
const struct clk_corediv_soc_desc *soc_desc = corediv->soc_desc;
const struct clk_corediv_desc *desc = corediv->desc; const struct clk_corediv_desc *desc = corediv->desc;
unsigned long flags = 0; unsigned long flags = 0;
u32 reg; u32 reg;
...@@ -97,7 +113,7 @@ static void clk_corediv_disable(struct clk_hw *hwclk) ...@@ -97,7 +113,7 @@ static void clk_corediv_disable(struct clk_hw *hwclk)
spin_lock_irqsave(&corediv->lock, flags); spin_lock_irqsave(&corediv->lock, flags);
reg = readl(corediv->reg); reg = readl(corediv->reg);
reg &= ~(BIT(desc->fieldbit) << CORE_CLK_DIV_ENABLE_OFFSET); reg &= ~(BIT(desc->fieldbit) << soc_desc->enable_bit_offset);
writel(reg, corediv->reg); writel(reg, corediv->reg);
spin_unlock_irqrestore(&corediv->lock, flags); spin_unlock_irqrestore(&corediv->lock, flags);
...@@ -107,10 +123,11 @@ static unsigned long clk_corediv_recalc_rate(struct clk_hw *hwclk, ...@@ -107,10 +123,11 @@ static unsigned long clk_corediv_recalc_rate(struct clk_hw *hwclk,
unsigned long parent_rate) unsigned long parent_rate)
{ {
struct clk_corediv *corediv = to_corediv_clk(hwclk); struct clk_corediv *corediv = to_corediv_clk(hwclk);
const struct clk_corediv_soc_desc *soc_desc = corediv->soc_desc;
const struct clk_corediv_desc *desc = corediv->desc; const struct clk_corediv_desc *desc = corediv->desc;
u32 reg, div; u32 reg, div;
reg = readl(corediv->reg + CORE_CLK_DIV_RATIO_OFFSET); reg = readl(corediv->reg + soc_desc->ratio_offset);
div = (reg >> desc->offset) & desc->mask; div = (reg >> desc->offset) & desc->mask;
return parent_rate / div; return parent_rate / div;
} }
...@@ -134,6 +151,7 @@ static int clk_corediv_set_rate(struct clk_hw *hwclk, unsigned long rate, ...@@ -134,6 +151,7 @@ static int clk_corediv_set_rate(struct clk_hw *hwclk, unsigned long rate,
unsigned long parent_rate) unsigned long parent_rate)
{ {
struct clk_corediv *corediv = to_corediv_clk(hwclk); struct clk_corediv *corediv = to_corediv_clk(hwclk);
const struct clk_corediv_soc_desc *soc_desc = corediv->soc_desc;
const struct clk_corediv_desc *desc = corediv->desc; const struct clk_corediv_desc *desc = corediv->desc;
unsigned long flags = 0; unsigned long flags = 0;
u32 reg, div; u32 reg, div;
...@@ -143,17 +161,17 @@ static int clk_corediv_set_rate(struct clk_hw *hwclk, unsigned long rate, ...@@ -143,17 +161,17 @@ static int clk_corediv_set_rate(struct clk_hw *hwclk, unsigned long rate,
spin_lock_irqsave(&corediv->lock, flags); spin_lock_irqsave(&corediv->lock, flags);
/* Write new divider to the divider ratio register */ /* Write new divider to the divider ratio register */
reg = readl(corediv->reg + CORE_CLK_DIV_RATIO_OFFSET); reg = readl(corediv->reg + soc_desc->ratio_offset);
reg &= ~(desc->mask << desc->offset); reg &= ~(desc->mask << desc->offset);
reg |= (div & desc->mask) << desc->offset; reg |= (div & desc->mask) << desc->offset;
writel(reg, corediv->reg + CORE_CLK_DIV_RATIO_OFFSET); writel(reg, corediv->reg + soc_desc->ratio_offset);
/* Set reload-force for this clock */ /* Set reload-force for this clock */
reg = readl(corediv->reg) | BIT(desc->fieldbit); reg = readl(corediv->reg) | BIT(desc->fieldbit);
writel(reg, corediv->reg); writel(reg, corediv->reg);
/* Now trigger the clock update */ /* Now trigger the clock update */
reg = readl(corediv->reg) | CORE_CLK_DIV_RATIO_RELOAD; reg = readl(corediv->reg) | soc_desc->ratio_reload;
writel(reg, corediv->reg); writel(reg, corediv->reg);
/* /*
...@@ -161,7 +179,7 @@ static int clk_corediv_set_rate(struct clk_hw *hwclk, unsigned long rate, ...@@ -161,7 +179,7 @@ static int clk_corediv_set_rate(struct clk_hw *hwclk, unsigned long rate,
* ratios request and the reload request. * ratios request and the reload request.
*/ */
udelay(1000); udelay(1000);
reg &= ~(CORE_CLK_DIV_RATIO_MASK | CORE_CLK_DIV_RATIO_RELOAD); reg &= ~(CORE_CLK_DIV_RATIO_MASK | soc_desc->ratio_reload);
writel(reg, corediv->reg); writel(reg, corediv->reg);
udelay(1000); udelay(1000);
...@@ -170,16 +188,25 @@ static int clk_corediv_set_rate(struct clk_hw *hwclk, unsigned long rate, ...@@ -170,16 +188,25 @@ static int clk_corediv_set_rate(struct clk_hw *hwclk, unsigned long rate,
return 0; return 0;
} }
static const struct clk_ops corediv_ops = { static const struct clk_corediv_soc_desc armada370_corediv_soc = {
.descs = mvebu_corediv_desc,
.ndescs = ARRAY_SIZE(mvebu_corediv_desc),
.ops = {
.enable = clk_corediv_enable, .enable = clk_corediv_enable,
.disable = clk_corediv_disable, .disable = clk_corediv_disable,
.is_enabled = clk_corediv_is_enabled, .is_enabled = clk_corediv_is_enabled,
.recalc_rate = clk_corediv_recalc_rate, .recalc_rate = clk_corediv_recalc_rate,
.round_rate = clk_corediv_round_rate, .round_rate = clk_corediv_round_rate,
.set_rate = clk_corediv_set_rate, .set_rate = clk_corediv_set_rate,
},
.ratio_reload = BIT(8),
.enable_bit_offset = 24,
.ratio_offset = 0x8,
}; };
static void __init mvebu_corediv_clk_init(struct device_node *node) static void __init
mvebu_corediv_clk_init(struct device_node *node,
const struct clk_corediv_soc_desc *soc_desc)
{ {
struct clk_init_data init; struct clk_init_data init;
struct clk_corediv *corediv; struct clk_corediv *corediv;
...@@ -195,7 +222,7 @@ static void __init mvebu_corediv_clk_init(struct device_node *node) ...@@ -195,7 +222,7 @@ static void __init mvebu_corediv_clk_init(struct device_node *node)
parent_name = of_clk_get_parent_name(node, 0); parent_name = of_clk_get_parent_name(node, 0);
clk_data.clk_num = ARRAY_SIZE(mvebu_corediv_desc); clk_data.clk_num = soc_desc->ndescs;
/* clks holds the clock array */ /* clks holds the clock array */
clks = kcalloc(clk_data.clk_num, sizeof(struct clk *), clks = kcalloc(clk_data.clk_num, sizeof(struct clk *),
...@@ -216,10 +243,11 @@ static void __init mvebu_corediv_clk_init(struct device_node *node) ...@@ -216,10 +243,11 @@ static void __init mvebu_corediv_clk_init(struct device_node *node)
init.num_parents = 1; init.num_parents = 1;
init.parent_names = &parent_name; init.parent_names = &parent_name;
init.name = clk_name; init.name = clk_name;
init.ops = &corediv_ops; init.ops = &soc_desc->ops;
init.flags = 0; init.flags = 0;
corediv[i].desc = mvebu_corediv_desc + i; corediv[i].soc_desc = soc_desc;
corediv[i].desc = soc_desc->descs + i;
corediv[i].reg = base; corediv[i].reg = base;
corediv[i].hw.init = &init; corediv[i].hw.init = &init;
...@@ -236,5 +264,10 @@ static void __init mvebu_corediv_clk_init(struct device_node *node) ...@@ -236,5 +264,10 @@ static void __init mvebu_corediv_clk_init(struct device_node *node)
err_unmap: err_unmap:
iounmap(base); iounmap(base);
} }
CLK_OF_DECLARE(mvebu_corediv_clk, "marvell,armada-370-corediv-clock",
mvebu_corediv_clk_init); static void __init armada370_corediv_clk_init(struct device_node *node)
{
return mvebu_corediv_clk_init(node, &armada370_corediv_soc);
}
CLK_OF_DECLARE(armada370_corediv_clk, "marvell,armada-370-corediv-clock",
armada370_corediv_clk_init);
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment