Commit ba52f8a9 authored by Soren Brinkmann's avatar Soren Brinkmann Committed by Michal Simek

clk/zynq/clkc: Add 'fclk-enable' feature

In some use cases Zynq's FPGA clocks are used as static clock
generators for IP in the FPGA part of the SOC for which no Linux driver
exists and would control those clocks. To avoid automatic
gating of these clocks in such cases a new property - fclk-enable - is
added to the clock controller's DT description to accomodate such use
cases. It's value is a bitmask, where a set bit results in enabling
the corresponding FCLK through the clkc.

FPGA clocks are handled following the rules below:

If an FCLK is not enabled by bootloaders, that FCLK will be disabled in
Linux. Drivers can enable and control it through the CCF as usual.

If an FCLK is enabled by bootloaders AND the corresponding bit in the
'fclk-enable' DT property is set, that FCLK will be enabled by the clkc,
resulting in an off by one reference count for that clock. Ensuring it
will always be running.
Signed-off-by: default avatarSoren Brinkmann <soren.brinkmann@xilinx.com>
Acked-by: default avatarMichal Simek <michal.simek@xilinx.com>
Signed-off-by: default avatarMichal Simek <michal.simek@xilinx.com>
parent 24c039f6
...@@ -22,6 +22,10 @@ Required properties: ...@@ -22,6 +22,10 @@ Required properties:
Optional properties: Optional properties:
- clocks : as described in the clock bindings - clocks : as described in the clock bindings
- clock-names : as described in the clock bindings - clock-names : as described in the clock bindings
- fclk-enable : Bit mask to enable FCLKs statically at boot time.
Bit [0..3] correspond to FCLK0..FCLK3. The corresponding
FCLK will only be enabled if it is actually running at
boot time.
Clock inputs: Clock inputs:
The following strings are optional parameters to the 'clock-names' property in The following strings are optional parameters to the 'clock-names' property in
......
...@@ -102,9 +102,10 @@ static const char *swdt_ext_clk_input_names[] __initdata = {"swdt_ext_clk"}; ...@@ -102,9 +102,10 @@ static const char *swdt_ext_clk_input_names[] __initdata = {"swdt_ext_clk"};
static void __init zynq_clk_register_fclk(enum zynq_clk fclk, static void __init zynq_clk_register_fclk(enum zynq_clk fclk,
const char *clk_name, void __iomem *fclk_ctrl_reg, const char *clk_name, void __iomem *fclk_ctrl_reg,
const char **parents) const char **parents, int enable)
{ {
struct clk *clk; struct clk *clk;
u32 enable_reg;
char *mux_name; char *mux_name;
char *div0_name; char *div0_name;
char *div1_name; char *div1_name;
...@@ -147,6 +148,12 @@ static void __init zynq_clk_register_fclk(enum zynq_clk fclk, ...@@ -147,6 +148,12 @@ static void __init zynq_clk_register_fclk(enum zynq_clk fclk,
clks[fclk] = clk_register_gate(NULL, clk_name, clks[fclk] = clk_register_gate(NULL, clk_name,
div1_name, CLK_SET_RATE_PARENT, fclk_gate_reg, div1_name, CLK_SET_RATE_PARENT, fclk_gate_reg,
0, CLK_GATE_SET_TO_DISABLE, fclk_gate_lock); 0, CLK_GATE_SET_TO_DISABLE, fclk_gate_lock);
enable_reg = readl(fclk_gate_reg) & 1;
if (enable && !enable_reg) {
if (clk_prepare_enable(clks[fclk]))
pr_warn("%s: FCLK%u enable failed\n", __func__,
fclk - fclk0);
}
kfree(mux_name); kfree(mux_name);
kfree(div0_name); kfree(div0_name);
kfree(div1_name); kfree(div1_name);
...@@ -213,6 +220,7 @@ static void __init zynq_clk_setup(struct device_node *np) ...@@ -213,6 +220,7 @@ static void __init zynq_clk_setup(struct device_node *np)
int ret; int ret;
struct clk *clk; struct clk *clk;
char *clk_name; char *clk_name;
unsigned int fclk_enable = 0;
const char *clk_output_name[clk_max]; const char *clk_output_name[clk_max];
const char *cpu_parents[4]; const char *cpu_parents[4];
const char *periph_parents[4]; const char *periph_parents[4];
...@@ -238,6 +246,8 @@ static void __init zynq_clk_setup(struct device_node *np) ...@@ -238,6 +246,8 @@ static void __init zynq_clk_setup(struct device_node *np)
periph_parents[2] = clk_output_name[armpll]; periph_parents[2] = clk_output_name[armpll];
periph_parents[3] = clk_output_name[ddrpll]; periph_parents[3] = clk_output_name[ddrpll];
of_property_read_u32(np, "fclk-enable", &fclk_enable);
/* ps_clk */ /* ps_clk */
ret = of_property_read_u32(np, "ps-clk-frequency", &tmp); ret = of_property_read_u32(np, "ps-clk-frequency", &tmp);
if (ret) { if (ret) {
...@@ -340,10 +350,12 @@ static void __init zynq_clk_setup(struct device_node *np) ...@@ -340,10 +350,12 @@ static void __init zynq_clk_setup(struct device_node *np)
clk_prepare_enable(clks[dci]); clk_prepare_enable(clks[dci]);
/* Peripheral clocks */ /* Peripheral clocks */
for (i = fclk0; i <= fclk3; i++) for (i = fclk0; i <= fclk3; i++) {
int enable = !!(fclk_enable & BIT(i - fclk0));
zynq_clk_register_fclk(i, clk_output_name[i], zynq_clk_register_fclk(i, clk_output_name[i],
SLCR_FPGA0_CLK_CTRL + 0x10 * (i - fclk0), SLCR_FPGA0_CLK_CTRL + 0x10 * (i - fclk0),
periph_parents); periph_parents, enable);
}
zynq_clk_register_periph_clk(lqspi, 0, clk_output_name[lqspi], NULL, zynq_clk_register_periph_clk(lqspi, 0, clk_output_name[lqspi], NULL,
SLCR_LQSPI_CLK_CTRL, periph_parents, 0); SLCR_LQSPI_CLK_CTRL, periph_parents, 0);
......
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