mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-11-14 15:54:15 +08:00
4e907ef6bd
The change in the clk API to return a per-user clock instance, moved
the clock state to struct clk_core so now the struct clk_hw .core field
is used instead of .clk for most operations.
So for hardware clocks that needs to share the same clock state, both
the .core and .clk pointers have to be assigned but currently only the
.clk is set. This leads to NULL pointer dereference when the operations
try to access the hw clock .core. For example, the composite clock rate
and mux components didn't have a .core set which leads to this error:
Unable to handle kernel NULL pointer dereference at virtual address 00000034
pgd = c0004000
[00000034] *pgd=00000000
Internal error: Oops: 5 [#1] PREEMPT SMP ARM
Modules linked in:
CPU: 0 PID: 1 Comm: swapper/0 Not tainted 3.19.0-next-20150211-00002-g1fb7f0e1150d #423
Hardware name: SAMSUNG EXYNOS (Flattened Device Tree)
task: ee480000 ti: ee488000 task.ti: ee488000
PC is at clk_mux_determine_rate_flags+0x14/0x19c
LR is at __clk_mux_determine_rate+0x24/0x2c
pc : [<c03a355c>] lr : [<c03a3734>] psr: a0000113
sp : ee489ce8 ip : ee489d84 fp : ee489d84
r10: 0000005c r9 : 00000001 r8 : 016e3600
r7 : 00000000 r6 : 00000000 r5 : ee442200 r4 : ee440c98
r3 : ffffffff r2 : 00000000 r1 : 016e3600 r0 : ee440c98
Flags: NzCv IRQs on FIQs on Mode SVC_32 ISA ARM Segment kernel
Control: 10c5387d Table: 4000406a DAC: 00000015
Process swapper/0 (pid: 1, stack limit = 0xee488210)
Stack: (0xee489ce8 to 0xee48a000)
9ce0: 00000000 ffffffff 60000113 ee440c98 ee442200 00000000
9d00: 016e3600 ffffffff 00000001 0000005c ee489d84 c03a3734 ee489d80 ee489d84
9d20: 00000000 c048b130 00000400 c03a5798 ee489d80 ee489d84 c0607f60 ffffffea
9d40: 00000001 00000001 ee489d5c c003f844 c06e3340 ee402680 ee440d0c ed935000
9d60: 016e3600 00000003 00000001 0000005c eded3700 c03a11a0 ee489d80 ee489d84
9d80: 016e3600 ee402680 c05b413a eddc9900 016e3600 c03a1228 00000000 ffffffff
9da0: ffffffff eddc9900 016e3600 c03a1c1c ffffffff 016e3600 ed8c6710 c03d6ce4
9dc0: eded3400 00000000 00000000 c03c797c 00000001 0000005c eded3700 eded3700
9de0: 000005e0 00000001 0000005c c03db8ac c06e7e54 c03c8f08 00000000 c06e7e64
9e00: c06b6e74 c06e7f64 000005e0 c06e7df8 c06e5100 00000000 c06e7e6c c06e7f54
9e20: 00000000 00000000 eebd9550 00000000 c06e7da0 c06e7e54 ee7b5010 c06e7da0
9e40: eddc9690 c06e7db4 c06b6e74 00000097 00000000 c03d4398 00000000 ee7b5010
9e60: eebd9550 c06e7da0 00000000 c03db824 ee7b5010 fffffffe c06e7db4 c0299c7c
9e80: ee7b5010 c072a05c 00000000 c0298858 ee7b5010 c06e7db4 ee7b5044 00000000
9ea0: eddc9580 c0298a04 c06e7db4 00000000 c0298978 c02971d4 ee405c78 ee732b40
9ec0: c06e7db4 eded3800 c06d6738 c0298044 c0608300 c06e7db4 00000000 c06e7db4
9ee0: 00000000 c06beb58 c06beb58 c0299024 00000000 c068dd00 00000000 c0008944
9f00: 00000038 c049013c ee462200 c0711920 ee480000 60000113 c06c2cb0 00000000
9f20: 00000000 c06c2cb0 60000113 00000000 ef7fcafc 00000000 c0640194 c00389ec
9f40: c05ec3a8 c063f824 00000006 00000006 c06c2c50 c0696444 00000006 c0696424
9f60: c06ee1c0 c066b588 c06b6e74 00000097 00000000 c066bd44 00000006 00000006
9f80: c066b588 c003d684 00000000 c0481938 00000000 00000000 00000000 00000000
9fa0: 00000000 c0481940 00000000 c000e680 00000000 00000000 00000000 00000000
9fc0: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 00000000
9fe0: 00000000 00000000 00000000 00000000 00000013 00000000 00000000 00000000
[<c03a355c>] (clk_mux_determine_rate_flags) from [<c03a3734>] (__clk_mux_determine_rate+0x24/0x2c)
[<c03a3734>] (__clk_mux_determine_rate) from [<c03a5798>] (clk_composite_determine_rate+0xbc/0x238)
[<c03a5798>] (clk_composite_determine_rate) from [<c03a11a0>] (clk_core_round_rate_nolock+0x5c/0x9c)
[<c03a11a0>] (clk_core_round_rate_nolock) from [<c03a1228>] (__clk_round_rate+0x38/0x40)
[<c03a1228>] (__clk_round_rate) from [<c03a1c1c>] (clk_round_rate+0x20/0x38)
[<c03a1c1c>] (clk_round_rate) from [<c03d6ce4>] (max98090_dai_set_sysclk+0x34/0x118)
[<c03d6ce4>] (max98090_dai_set_sysclk) from [<c03c797c>] (snd_soc_dai_set_sysclk+0x38/0x80)
[<c03c797c>] (snd_soc_dai_set_sysclk) from [<c03db8ac>] (snow_late_probe+0x24/0x48)
[<c03db8ac>] (snow_late_probe) from [<c03c8f08>] (snd_soc_register_card+0xf04/0x1070)
[<c03c8f08>] (snd_soc_register_card) from [<c03d4398>] (devm_snd_soc_register_card+0x30/0x64)
[<c03d4398>] (devm_snd_soc_register_card) from [<c03db824>] (snow_probe+0x68/0xcc)
[<c03db824>] (snow_probe) from [<c0299c7c>] (platform_drv_probe+0x48/0x98)
[<c0299c7c>] (platform_drv_probe) from [<c0298858>] (driver_probe_device+0x114/0x234)
[<c0298858>] (driver_probe_device) from [<c0298a04>] (__driver_attach+0x8c/0x90)
[<c0298a04>] (__driver_attach) from [<c02971d4>] (bus_for_each_dev+0x54/0x88)
[<c02971d4>] (bus_for_each_dev) from [<c0298044>] (bus_add_driver+0xd8/0x1cc)
[<c0298044>] (bus_add_driver) from [<c0299024>] (driver_register+0x78/0xf4)
[<c0299024>] (driver_register) from [<c0008944>] (do_one_initcall+0x80/0x1d0)
[<c0008944>] (do_one_initcall) from [<c066bd44>] (kernel_init_freeable+0x10c/0x1d8)
[<c066bd44>] (kernel_init_freeable) from [<c0481940>] (kernel_init+0x8/0xe4)
[<c0481940>] (kernel_init) from [<c000e680>] (ret_from_fork+0x14/0x34)
Code: e24dd00c e5907000 e1a08001 e88d000c (e5970034)
The changes were made using the following cocinelle semantic patch:
@i@
@@
@depends on i@
identifier dst;
@@
- dst->clk = hw->clk;
+ __clk_hw_set_clk(dst, hw);
@depends on i@
identifier dst;
@@
- dst->hw.clk = hw->clk;
+ __clk_hw_set_clk(&dst->hw, hw);
Fixes: 035a61c314
("clk: Make clk API return per-user struct clk instances")
Signed-off-by: Javier Martinez Canillas <javier.martinez@collabora.co.uk>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Michael Turquette <mturquette@linaro.org>
831 lines
19 KiB
C
831 lines
19 KiB
C
/*
|
|
* clkgen-mux.c: ST GEN-MUX Clock driver
|
|
*
|
|
* Copyright (C) 2014 STMicroelectronics (R&D) Limited
|
|
*
|
|
* Authors: Stephen Gallimore <stephen.gallimore@st.com>
|
|
* Pankaj Dev <pankaj.dev@st.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
*/
|
|
|
|
#include <linux/slab.h>
|
|
#include <linux/of_address.h>
|
|
#include <linux/clk-provider.h>
|
|
|
|
static DEFINE_SPINLOCK(clkgena_divmux_lock);
|
|
static DEFINE_SPINLOCK(clkgenf_lock);
|
|
|
|
static const char ** __init clkgen_mux_get_parents(struct device_node *np,
|
|
int *num_parents)
|
|
{
|
|
const char **parents;
|
|
int nparents, i;
|
|
|
|
nparents = of_count_phandle_with_args(np, "clocks", "#clock-cells");
|
|
if (WARN_ON(nparents <= 0))
|
|
return ERR_PTR(-EINVAL);
|
|
|
|
parents = kzalloc(nparents * sizeof(const char *), GFP_KERNEL);
|
|
if (!parents)
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
for (i = 0; i < nparents; i++)
|
|
parents[i] = of_clk_get_parent_name(np, i);
|
|
|
|
*num_parents = nparents;
|
|
return parents;
|
|
}
|
|
|
|
/**
|
|
* DOC: Clock mux with a programmable divider on each of its three inputs.
|
|
* The mux has an input setting which effectively gates its output.
|
|
*
|
|
* Traits of this clock:
|
|
* prepare - clk_(un)prepare only ensures parent is (un)prepared
|
|
* enable - clk_enable and clk_disable are functional & control gating
|
|
* rate - set rate is supported
|
|
* parent - set/get parent
|
|
*/
|
|
|
|
#define NUM_INPUTS 3
|
|
|
|
struct clkgena_divmux {
|
|
struct clk_hw hw;
|
|
/* Subclassed mux and divider structures */
|
|
struct clk_mux mux;
|
|
struct clk_divider div[NUM_INPUTS];
|
|
/* Enable/running feedback register bits for each input */
|
|
void __iomem *feedback_reg[NUM_INPUTS];
|
|
int feedback_bit_idx;
|
|
|
|
u8 muxsel;
|
|
};
|
|
|
|
#define to_clkgena_divmux(_hw) container_of(_hw, struct clkgena_divmux, hw)
|
|
|
|
struct clkgena_divmux_data {
|
|
int num_outputs;
|
|
int mux_offset;
|
|
int mux_offset2;
|
|
int mux_start_bit;
|
|
int div_offsets[NUM_INPUTS];
|
|
int fb_offsets[NUM_INPUTS];
|
|
int fb_start_bit_idx;
|
|
};
|
|
|
|
#define CKGAX_CLKOPSRC_SWITCH_OFF 0x3
|
|
|
|
static int clkgena_divmux_is_running(struct clkgena_divmux *mux)
|
|
{
|
|
u32 regval = readl(mux->feedback_reg[mux->muxsel]);
|
|
u32 running = regval & BIT(mux->feedback_bit_idx);
|
|
return !!running;
|
|
}
|
|
|
|
static int clkgena_divmux_enable(struct clk_hw *hw)
|
|
{
|
|
struct clkgena_divmux *genamux = to_clkgena_divmux(hw);
|
|
struct clk_hw *mux_hw = &genamux->mux.hw;
|
|
unsigned long timeout;
|
|
int ret = 0;
|
|
|
|
__clk_hw_set_clk(mux_hw, hw);
|
|
|
|
ret = clk_mux_ops.set_parent(mux_hw, genamux->muxsel);
|
|
if (ret)
|
|
return ret;
|
|
|
|
timeout = jiffies + msecs_to_jiffies(10);
|
|
|
|
while (!clkgena_divmux_is_running(genamux)) {
|
|
if (time_after(jiffies, timeout))
|
|
return -ETIMEDOUT;
|
|
cpu_relax();
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void clkgena_divmux_disable(struct clk_hw *hw)
|
|
{
|
|
struct clkgena_divmux *genamux = to_clkgena_divmux(hw);
|
|
struct clk_hw *mux_hw = &genamux->mux.hw;
|
|
|
|
__clk_hw_set_clk(mux_hw, hw);
|
|
|
|
clk_mux_ops.set_parent(mux_hw, CKGAX_CLKOPSRC_SWITCH_OFF);
|
|
}
|
|
|
|
static int clkgena_divmux_is_enabled(struct clk_hw *hw)
|
|
{
|
|
struct clkgena_divmux *genamux = to_clkgena_divmux(hw);
|
|
struct clk_hw *mux_hw = &genamux->mux.hw;
|
|
|
|
__clk_hw_set_clk(mux_hw, hw);
|
|
|
|
return (s8)clk_mux_ops.get_parent(mux_hw) > 0;
|
|
}
|
|
|
|
u8 clkgena_divmux_get_parent(struct clk_hw *hw)
|
|
{
|
|
struct clkgena_divmux *genamux = to_clkgena_divmux(hw);
|
|
struct clk_hw *mux_hw = &genamux->mux.hw;
|
|
|
|
__clk_hw_set_clk(mux_hw, hw);
|
|
|
|
genamux->muxsel = clk_mux_ops.get_parent(mux_hw);
|
|
if ((s8)genamux->muxsel < 0) {
|
|
pr_debug("%s: %s: Invalid parent, setting to default.\n",
|
|
__func__, __clk_get_name(hw->clk));
|
|
genamux->muxsel = 0;
|
|
}
|
|
|
|
return genamux->muxsel;
|
|
}
|
|
|
|
static int clkgena_divmux_set_parent(struct clk_hw *hw, u8 index)
|
|
{
|
|
struct clkgena_divmux *genamux = to_clkgena_divmux(hw);
|
|
|
|
if (index >= CKGAX_CLKOPSRC_SWITCH_OFF)
|
|
return -EINVAL;
|
|
|
|
genamux->muxsel = index;
|
|
|
|
/*
|
|
* If the mux is already enabled, call enable directly to set the
|
|
* new mux position and wait for it to start running again. Otherwise
|
|
* do nothing.
|
|
*/
|
|
if (clkgena_divmux_is_enabled(hw))
|
|
clkgena_divmux_enable(hw);
|
|
|
|
return 0;
|
|
}
|
|
|
|
unsigned long clkgena_divmux_recalc_rate(struct clk_hw *hw,
|
|
unsigned long parent_rate)
|
|
{
|
|
struct clkgena_divmux *genamux = to_clkgena_divmux(hw);
|
|
struct clk_hw *div_hw = &genamux->div[genamux->muxsel].hw;
|
|
|
|
__clk_hw_set_clk(div_hw, hw);
|
|
|
|
return clk_divider_ops.recalc_rate(div_hw, parent_rate);
|
|
}
|
|
|
|
static int clkgena_divmux_set_rate(struct clk_hw *hw, unsigned long rate,
|
|
unsigned long parent_rate)
|
|
{
|
|
struct clkgena_divmux *genamux = to_clkgena_divmux(hw);
|
|
struct clk_hw *div_hw = &genamux->div[genamux->muxsel].hw;
|
|
|
|
__clk_hw_set_clk(div_hw, hw);
|
|
|
|
return clk_divider_ops.set_rate(div_hw, rate, parent_rate);
|
|
}
|
|
|
|
static long clkgena_divmux_round_rate(struct clk_hw *hw, unsigned long rate,
|
|
unsigned long *prate)
|
|
{
|
|
struct clkgena_divmux *genamux = to_clkgena_divmux(hw);
|
|
struct clk_hw *div_hw = &genamux->div[genamux->muxsel].hw;
|
|
|
|
__clk_hw_set_clk(div_hw, hw);
|
|
|
|
return clk_divider_ops.round_rate(div_hw, rate, prate);
|
|
}
|
|
|
|
static const struct clk_ops clkgena_divmux_ops = {
|
|
.enable = clkgena_divmux_enable,
|
|
.disable = clkgena_divmux_disable,
|
|
.is_enabled = clkgena_divmux_is_enabled,
|
|
.get_parent = clkgena_divmux_get_parent,
|
|
.set_parent = clkgena_divmux_set_parent,
|
|
.round_rate = clkgena_divmux_round_rate,
|
|
.recalc_rate = clkgena_divmux_recalc_rate,
|
|
.set_rate = clkgena_divmux_set_rate,
|
|
};
|
|
|
|
/**
|
|
* clk_register_genamux - register a genamux clock with the clock framework
|
|
*/
|
|
struct clk *clk_register_genamux(const char *name,
|
|
const char **parent_names, u8 num_parents,
|
|
void __iomem *reg,
|
|
const struct clkgena_divmux_data *muxdata,
|
|
u32 idx)
|
|
{
|
|
/*
|
|
* Fixed constants across all ClockgenA variants
|
|
*/
|
|
const int mux_width = 2;
|
|
const int divider_width = 5;
|
|
struct clkgena_divmux *genamux;
|
|
struct clk *clk;
|
|
struct clk_init_data init;
|
|
int i;
|
|
|
|
genamux = kzalloc(sizeof(*genamux), GFP_KERNEL);
|
|
if (!genamux)
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
init.name = name;
|
|
init.ops = &clkgena_divmux_ops;
|
|
init.flags = CLK_IS_BASIC;
|
|
init.parent_names = parent_names;
|
|
init.num_parents = num_parents;
|
|
|
|
genamux->mux.lock = &clkgena_divmux_lock;
|
|
genamux->mux.mask = BIT(mux_width) - 1;
|
|
genamux->mux.shift = muxdata->mux_start_bit + (idx * mux_width);
|
|
if (genamux->mux.shift > 31) {
|
|
/*
|
|
* We have spilled into the second mux register so
|
|
* adjust the register address and the bit shift accordingly
|
|
*/
|
|
genamux->mux.reg = reg + muxdata->mux_offset2;
|
|
genamux->mux.shift -= 32;
|
|
} else {
|
|
genamux->mux.reg = reg + muxdata->mux_offset;
|
|
}
|
|
|
|
for (i = 0; i < NUM_INPUTS; i++) {
|
|
/*
|
|
* Divider config for each input
|
|
*/
|
|
void __iomem *divbase = reg + muxdata->div_offsets[i];
|
|
genamux->div[i].width = divider_width;
|
|
genamux->div[i].reg = divbase + (idx * sizeof(u32));
|
|
|
|
/*
|
|
* Mux enabled/running feedback register for each input.
|
|
*/
|
|
genamux->feedback_reg[i] = reg + muxdata->fb_offsets[i];
|
|
}
|
|
|
|
genamux->feedback_bit_idx = muxdata->fb_start_bit_idx + idx;
|
|
genamux->hw.init = &init;
|
|
|
|
clk = clk_register(NULL, &genamux->hw);
|
|
if (IS_ERR(clk)) {
|
|
kfree(genamux);
|
|
goto err;
|
|
}
|
|
|
|
pr_debug("%s: parent %s rate %lu\n",
|
|
__clk_get_name(clk),
|
|
__clk_get_name(clk_get_parent(clk)),
|
|
clk_get_rate(clk));
|
|
err:
|
|
return clk;
|
|
}
|
|
|
|
static struct clkgena_divmux_data st_divmux_c65hs = {
|
|
.num_outputs = 4,
|
|
.mux_offset = 0x14,
|
|
.mux_start_bit = 0,
|
|
.div_offsets = { 0x800, 0x900, 0xb00 },
|
|
.fb_offsets = { 0x18, 0x1c, 0x20 },
|
|
.fb_start_bit_idx = 0,
|
|
};
|
|
|
|
static struct clkgena_divmux_data st_divmux_c65ls = {
|
|
.num_outputs = 14,
|
|
.mux_offset = 0x14,
|
|
.mux_offset2 = 0x24,
|
|
.mux_start_bit = 8,
|
|
.div_offsets = { 0x810, 0xa10, 0xb10 },
|
|
.fb_offsets = { 0x18, 0x1c, 0x20 },
|
|
.fb_start_bit_idx = 4,
|
|
};
|
|
|
|
static struct clkgena_divmux_data st_divmux_c32odf0 = {
|
|
.num_outputs = 8,
|
|
.mux_offset = 0x1c,
|
|
.mux_start_bit = 0,
|
|
.div_offsets = { 0x800, 0x900, 0xa60 },
|
|
.fb_offsets = { 0x2c, 0x24, 0x28 },
|
|
.fb_start_bit_idx = 0,
|
|
};
|
|
|
|
static struct clkgena_divmux_data st_divmux_c32odf1 = {
|
|
.num_outputs = 8,
|
|
.mux_offset = 0x1c,
|
|
.mux_start_bit = 16,
|
|
.div_offsets = { 0x820, 0x980, 0xa80 },
|
|
.fb_offsets = { 0x2c, 0x24, 0x28 },
|
|
.fb_start_bit_idx = 8,
|
|
};
|
|
|
|
static struct clkgena_divmux_data st_divmux_c32odf2 = {
|
|
.num_outputs = 8,
|
|
.mux_offset = 0x20,
|
|
.mux_start_bit = 0,
|
|
.div_offsets = { 0x840, 0xa20, 0xb10 },
|
|
.fb_offsets = { 0x2c, 0x24, 0x28 },
|
|
.fb_start_bit_idx = 16,
|
|
};
|
|
|
|
static struct clkgena_divmux_data st_divmux_c32odf3 = {
|
|
.num_outputs = 8,
|
|
.mux_offset = 0x20,
|
|
.mux_start_bit = 16,
|
|
.div_offsets = { 0x860, 0xa40, 0xb30 },
|
|
.fb_offsets = { 0x2c, 0x24, 0x28 },
|
|
.fb_start_bit_idx = 24,
|
|
};
|
|
|
|
static struct of_device_id clkgena_divmux_of_match[] = {
|
|
{
|
|
.compatible = "st,clkgena-divmux-c65-hs",
|
|
.data = &st_divmux_c65hs,
|
|
},
|
|
{
|
|
.compatible = "st,clkgena-divmux-c65-ls",
|
|
.data = &st_divmux_c65ls,
|
|
},
|
|
{
|
|
.compatible = "st,clkgena-divmux-c32-odf0",
|
|
.data = &st_divmux_c32odf0,
|
|
},
|
|
{
|
|
.compatible = "st,clkgena-divmux-c32-odf1",
|
|
.data = &st_divmux_c32odf1,
|
|
},
|
|
{
|
|
.compatible = "st,clkgena-divmux-c32-odf2",
|
|
.data = &st_divmux_c32odf2,
|
|
},
|
|
{
|
|
.compatible = "st,clkgena-divmux-c32-odf3",
|
|
.data = &st_divmux_c32odf3,
|
|
},
|
|
{}
|
|
};
|
|
|
|
static void __iomem * __init clkgen_get_register_base(
|
|
struct device_node *np)
|
|
{
|
|
struct device_node *pnode;
|
|
void __iomem *reg = NULL;
|
|
|
|
pnode = of_get_parent(np);
|
|
if (!pnode)
|
|
return NULL;
|
|
|
|
reg = of_iomap(pnode, 0);
|
|
|
|
of_node_put(pnode);
|
|
return reg;
|
|
}
|
|
|
|
void __init st_of_clkgena_divmux_setup(struct device_node *np)
|
|
{
|
|
const struct of_device_id *match;
|
|
const struct clkgena_divmux_data *data;
|
|
struct clk_onecell_data *clk_data;
|
|
void __iomem *reg;
|
|
const char **parents;
|
|
int num_parents = 0, i;
|
|
|
|
match = of_match_node(clkgena_divmux_of_match, np);
|
|
if (WARN_ON(!match))
|
|
return;
|
|
|
|
data = (struct clkgena_divmux_data *)match->data;
|
|
|
|
reg = clkgen_get_register_base(np);
|
|
if (!reg)
|
|
return;
|
|
|
|
parents = clkgen_mux_get_parents(np, &num_parents);
|
|
if (IS_ERR(parents))
|
|
return;
|
|
|
|
clk_data = kzalloc(sizeof(*clk_data), GFP_KERNEL);
|
|
if (!clk_data)
|
|
goto err;
|
|
|
|
clk_data->clk_num = data->num_outputs;
|
|
clk_data->clks = kzalloc(clk_data->clk_num * sizeof(struct clk *),
|
|
GFP_KERNEL);
|
|
|
|
if (!clk_data->clks)
|
|
goto err;
|
|
|
|
for (i = 0; i < clk_data->clk_num; i++) {
|
|
struct clk *clk;
|
|
const char *clk_name;
|
|
|
|
if (of_property_read_string_index(np, "clock-output-names",
|
|
i, &clk_name))
|
|
break;
|
|
|
|
/*
|
|
* If we read an empty clock name then the output is unused
|
|
*/
|
|
if (*clk_name == '\0')
|
|
continue;
|
|
|
|
clk = clk_register_genamux(clk_name, parents, num_parents,
|
|
reg, data, i);
|
|
|
|
if (IS_ERR(clk))
|
|
goto err;
|
|
|
|
clk_data->clks[i] = clk;
|
|
}
|
|
|
|
kfree(parents);
|
|
|
|
of_clk_add_provider(np, of_clk_src_onecell_get, clk_data);
|
|
return;
|
|
err:
|
|
if (clk_data)
|
|
kfree(clk_data->clks);
|
|
|
|
kfree(clk_data);
|
|
kfree(parents);
|
|
}
|
|
CLK_OF_DECLARE(clkgenadivmux, "st,clkgena-divmux", st_of_clkgena_divmux_setup);
|
|
|
|
struct clkgena_prediv_data {
|
|
u32 offset;
|
|
u8 shift;
|
|
struct clk_div_table *table;
|
|
};
|
|
|
|
static struct clk_div_table prediv_table16[] = {
|
|
{ .val = 0, .div = 1 },
|
|
{ .val = 1, .div = 16 },
|
|
{ .div = 0 },
|
|
};
|
|
|
|
static struct clkgena_prediv_data prediv_c65_data = {
|
|
.offset = 0x4c,
|
|
.shift = 31,
|
|
.table = prediv_table16,
|
|
};
|
|
|
|
static struct clkgena_prediv_data prediv_c32_data = {
|
|
.offset = 0x50,
|
|
.shift = 1,
|
|
.table = prediv_table16,
|
|
};
|
|
|
|
static struct of_device_id clkgena_prediv_of_match[] = {
|
|
{ .compatible = "st,clkgena-prediv-c65", .data = &prediv_c65_data },
|
|
{ .compatible = "st,clkgena-prediv-c32", .data = &prediv_c32_data },
|
|
{}
|
|
};
|
|
|
|
void __init st_of_clkgena_prediv_setup(struct device_node *np)
|
|
{
|
|
const struct of_device_id *match;
|
|
void __iomem *reg;
|
|
const char *parent_name, *clk_name;
|
|
struct clk *clk;
|
|
struct clkgena_prediv_data *data;
|
|
|
|
match = of_match_node(clkgena_prediv_of_match, np);
|
|
if (!match) {
|
|
pr_err("%s: No matching data\n", __func__);
|
|
return;
|
|
}
|
|
|
|
data = (struct clkgena_prediv_data *)match->data;
|
|
|
|
reg = clkgen_get_register_base(np);
|
|
if (!reg)
|
|
return;
|
|
|
|
parent_name = of_clk_get_parent_name(np, 0);
|
|
if (!parent_name)
|
|
return;
|
|
|
|
if (of_property_read_string_index(np, "clock-output-names",
|
|
0, &clk_name))
|
|
return;
|
|
|
|
clk = clk_register_divider_table(NULL, clk_name, parent_name, 0,
|
|
reg + data->offset, data->shift, 1,
|
|
0, data->table, NULL);
|
|
if (IS_ERR(clk))
|
|
return;
|
|
|
|
of_clk_add_provider(np, of_clk_src_simple_get, clk);
|
|
pr_debug("%s: parent %s rate %u\n",
|
|
__clk_get_name(clk),
|
|
__clk_get_name(clk_get_parent(clk)),
|
|
(unsigned int)clk_get_rate(clk));
|
|
|
|
return;
|
|
}
|
|
CLK_OF_DECLARE(clkgenaprediv, "st,clkgena-prediv", st_of_clkgena_prediv_setup);
|
|
|
|
struct clkgen_mux_data {
|
|
u32 offset;
|
|
u8 shift;
|
|
u8 width;
|
|
spinlock_t *lock;
|
|
unsigned long clk_flags;
|
|
u8 mux_flags;
|
|
};
|
|
|
|
static struct clkgen_mux_data clkgen_mux_c_vcc_hd_416 = {
|
|
.offset = 0,
|
|
.shift = 0,
|
|
.width = 1,
|
|
};
|
|
|
|
static struct clkgen_mux_data clkgen_mux_f_vcc_fvdp_416 = {
|
|
.offset = 0,
|
|
.shift = 0,
|
|
.width = 1,
|
|
};
|
|
|
|
static struct clkgen_mux_data clkgen_mux_f_vcc_hva_416 = {
|
|
.offset = 0,
|
|
.shift = 0,
|
|
.width = 1,
|
|
};
|
|
|
|
static struct clkgen_mux_data clkgen_mux_f_vcc_hd_416 = {
|
|
.offset = 0,
|
|
.shift = 16,
|
|
.width = 1,
|
|
.lock = &clkgenf_lock,
|
|
};
|
|
|
|
static struct clkgen_mux_data clkgen_mux_c_vcc_sd_416 = {
|
|
.offset = 0,
|
|
.shift = 17,
|
|
.width = 1,
|
|
.lock = &clkgenf_lock,
|
|
};
|
|
|
|
static struct clkgen_mux_data stih415_a9_mux_data = {
|
|
.offset = 0,
|
|
.shift = 1,
|
|
.width = 2,
|
|
};
|
|
static struct clkgen_mux_data stih416_a9_mux_data = {
|
|
.offset = 0,
|
|
.shift = 0,
|
|
.width = 2,
|
|
};
|
|
static struct clkgen_mux_data stih407_a9_mux_data = {
|
|
.offset = 0x1a4,
|
|
.shift = 1,
|
|
.width = 2,
|
|
};
|
|
|
|
static struct of_device_id mux_of_match[] = {
|
|
{
|
|
.compatible = "st,stih416-clkgenc-vcc-hd",
|
|
.data = &clkgen_mux_c_vcc_hd_416,
|
|
},
|
|
{
|
|
.compatible = "st,stih416-clkgenf-vcc-fvdp",
|
|
.data = &clkgen_mux_f_vcc_fvdp_416,
|
|
},
|
|
{
|
|
.compatible = "st,stih416-clkgenf-vcc-hva",
|
|
.data = &clkgen_mux_f_vcc_hva_416,
|
|
},
|
|
{
|
|
.compatible = "st,stih416-clkgenf-vcc-hd",
|
|
.data = &clkgen_mux_f_vcc_hd_416,
|
|
},
|
|
{
|
|
.compatible = "st,stih416-clkgenf-vcc-sd",
|
|
.data = &clkgen_mux_c_vcc_sd_416,
|
|
},
|
|
{
|
|
.compatible = "st,stih415-clkgen-a9-mux",
|
|
.data = &stih415_a9_mux_data,
|
|
},
|
|
{
|
|
.compatible = "st,stih416-clkgen-a9-mux",
|
|
.data = &stih416_a9_mux_data,
|
|
},
|
|
{
|
|
.compatible = "st,stih407-clkgen-a9-mux",
|
|
.data = &stih407_a9_mux_data,
|
|
},
|
|
{}
|
|
};
|
|
|
|
void __init st_of_clkgen_mux_setup(struct device_node *np)
|
|
{
|
|
const struct of_device_id *match;
|
|
struct clk *clk;
|
|
void __iomem *reg;
|
|
const char **parents;
|
|
int num_parents;
|
|
struct clkgen_mux_data *data;
|
|
|
|
match = of_match_node(mux_of_match, np);
|
|
if (!match) {
|
|
pr_err("%s: No matching data\n", __func__);
|
|
return;
|
|
}
|
|
|
|
data = (struct clkgen_mux_data *)match->data;
|
|
|
|
reg = of_iomap(np, 0);
|
|
if (!reg) {
|
|
pr_err("%s: Failed to get base address\n", __func__);
|
|
return;
|
|
}
|
|
|
|
parents = clkgen_mux_get_parents(np, &num_parents);
|
|
if (IS_ERR(parents)) {
|
|
pr_err("%s: Failed to get parents (%ld)\n",
|
|
__func__, PTR_ERR(parents));
|
|
return;
|
|
}
|
|
|
|
clk = clk_register_mux(NULL, np->name, parents, num_parents,
|
|
data->clk_flags | CLK_SET_RATE_PARENT,
|
|
reg + data->offset,
|
|
data->shift, data->width, data->mux_flags,
|
|
data->lock);
|
|
if (IS_ERR(clk))
|
|
goto err;
|
|
|
|
pr_debug("%s: parent %s rate %u\n",
|
|
__clk_get_name(clk),
|
|
__clk_get_name(clk_get_parent(clk)),
|
|
(unsigned int)clk_get_rate(clk));
|
|
|
|
of_clk_add_provider(np, of_clk_src_simple_get, clk);
|
|
|
|
err:
|
|
kfree(parents);
|
|
|
|
return;
|
|
}
|
|
CLK_OF_DECLARE(clkgen_mux, "st,clkgen-mux", st_of_clkgen_mux_setup);
|
|
|
|
#define VCC_MAX_CHANNELS 16
|
|
|
|
#define VCC_GATE_OFFSET 0x0
|
|
#define VCC_MUX_OFFSET 0x4
|
|
#define VCC_DIV_OFFSET 0x8
|
|
|
|
struct clkgen_vcc_data {
|
|
spinlock_t *lock;
|
|
unsigned long clk_flags;
|
|
};
|
|
|
|
static struct clkgen_vcc_data st_clkgenc_vcc_416 = {
|
|
.clk_flags = CLK_SET_RATE_PARENT,
|
|
};
|
|
|
|
static struct clkgen_vcc_data st_clkgenf_vcc_416 = {
|
|
.lock = &clkgenf_lock,
|
|
};
|
|
|
|
static struct of_device_id vcc_of_match[] = {
|
|
{ .compatible = "st,stih416-clkgenc", .data = &st_clkgenc_vcc_416 },
|
|
{ .compatible = "st,stih416-clkgenf", .data = &st_clkgenf_vcc_416 },
|
|
{}
|
|
};
|
|
|
|
void __init st_of_clkgen_vcc_setup(struct device_node *np)
|
|
{
|
|
const struct of_device_id *match;
|
|
void __iomem *reg;
|
|
const char **parents;
|
|
int num_parents, i;
|
|
struct clk_onecell_data *clk_data;
|
|
struct clkgen_vcc_data *data;
|
|
|
|
match = of_match_node(vcc_of_match, np);
|
|
if (WARN_ON(!match))
|
|
return;
|
|
data = (struct clkgen_vcc_data *)match->data;
|
|
|
|
reg = of_iomap(np, 0);
|
|
if (!reg)
|
|
return;
|
|
|
|
parents = clkgen_mux_get_parents(np, &num_parents);
|
|
if (IS_ERR(parents))
|
|
return;
|
|
|
|
clk_data = kzalloc(sizeof(*clk_data), GFP_KERNEL);
|
|
if (!clk_data)
|
|
goto err;
|
|
|
|
clk_data->clk_num = VCC_MAX_CHANNELS;
|
|
clk_data->clks = kzalloc(clk_data->clk_num * sizeof(struct clk *),
|
|
GFP_KERNEL);
|
|
|
|
if (!clk_data->clks)
|
|
goto err;
|
|
|
|
for (i = 0; i < clk_data->clk_num; i++) {
|
|
struct clk *clk;
|
|
const char *clk_name;
|
|
struct clk_gate *gate;
|
|
struct clk_divider *div;
|
|
struct clk_mux *mux;
|
|
|
|
if (of_property_read_string_index(np, "clock-output-names",
|
|
i, &clk_name))
|
|
break;
|
|
|
|
/*
|
|
* If we read an empty clock name then the output is unused
|
|
*/
|
|
if (*clk_name == '\0')
|
|
continue;
|
|
|
|
gate = kzalloc(sizeof(struct clk_gate), GFP_KERNEL);
|
|
if (!gate)
|
|
break;
|
|
|
|
div = kzalloc(sizeof(struct clk_divider), GFP_KERNEL);
|
|
if (!div) {
|
|
kfree(gate);
|
|
break;
|
|
}
|
|
|
|
mux = kzalloc(sizeof(struct clk_mux), GFP_KERNEL);
|
|
if (!mux) {
|
|
kfree(gate);
|
|
kfree(div);
|
|
break;
|
|
}
|
|
|
|
gate->reg = reg + VCC_GATE_OFFSET;
|
|
gate->bit_idx = i;
|
|
gate->flags = CLK_GATE_SET_TO_DISABLE;
|
|
gate->lock = data->lock;
|
|
|
|
div->reg = reg + VCC_DIV_OFFSET;
|
|
div->shift = 2 * i;
|
|
div->width = 2;
|
|
div->flags = CLK_DIVIDER_POWER_OF_TWO |
|
|
CLK_DIVIDER_ROUND_CLOSEST;
|
|
|
|
mux->reg = reg + VCC_MUX_OFFSET;
|
|
mux->shift = 2 * i;
|
|
mux->mask = 0x3;
|
|
|
|
clk = clk_register_composite(NULL, clk_name, parents,
|
|
num_parents,
|
|
&mux->hw, &clk_mux_ops,
|
|
&div->hw, &clk_divider_ops,
|
|
&gate->hw, &clk_gate_ops,
|
|
data->clk_flags);
|
|
if (IS_ERR(clk)) {
|
|
kfree(gate);
|
|
kfree(div);
|
|
kfree(mux);
|
|
goto err;
|
|
}
|
|
|
|
pr_debug("%s: parent %s rate %u\n",
|
|
__clk_get_name(clk),
|
|
__clk_get_name(clk_get_parent(clk)),
|
|
(unsigned int)clk_get_rate(clk));
|
|
|
|
clk_data->clks[i] = clk;
|
|
}
|
|
|
|
kfree(parents);
|
|
|
|
of_clk_add_provider(np, of_clk_src_onecell_get, clk_data);
|
|
return;
|
|
|
|
err:
|
|
for (i = 0; i < clk_data->clk_num; i++) {
|
|
struct clk_composite *composite;
|
|
|
|
if (!clk_data->clks[i])
|
|
continue;
|
|
|
|
composite = container_of(__clk_get_hw(clk_data->clks[i]),
|
|
struct clk_composite, hw);
|
|
kfree(container_of(composite->gate_hw, struct clk_gate, hw));
|
|
kfree(container_of(composite->rate_hw, struct clk_divider, hw));
|
|
kfree(container_of(composite->mux_hw, struct clk_mux, hw));
|
|
}
|
|
|
|
if (clk_data)
|
|
kfree(clk_data->clks);
|
|
|
|
kfree(clk_data);
|
|
kfree(parents);
|
|
}
|
|
CLK_OF_DECLARE(clkgen_vcc, "st,clkgen-vcc", st_of_clkgen_vcc_setup);
|