1 // SPDX-License-Identifier: GPL-2.0-only
3 * Copyright (C) 2015 Altera Corporation. All rights reserved
5 #include <linux/slab.h>
6 #include <linux/clk-provider.h>
12 #define CLK_MGR_FREE_SHIFT 16
13 #define CLK_MGR_FREE_MASK 0x7
15 #define SOCFPGA_MPU_FREE_CLK "mpu_free_clk"
16 #define SOCFPGA_NOC_FREE_CLK "noc_free_clk"
17 #define SOCFPGA_SDMMC_FREE_CLK "sdmmc_free_clk"
18 #define to_socfpga_periph_clk(p) container_of(p, struct socfpga_periph_clk, hw.hw)
20 static unsigned long clk_periclk_recalc_rate(struct clk_hw
*hwclk
,
21 unsigned long parent_rate
)
23 struct socfpga_periph_clk
*socfpgaclk
= to_socfpga_periph_clk(hwclk
);
26 if (socfpgaclk
->fixed_div
) {
27 div
= socfpgaclk
->fixed_div
;
28 } else if (socfpgaclk
->div_reg
) {
29 div
= readl(socfpgaclk
->div_reg
) >> socfpgaclk
->shift
;
30 div
&= GENMASK(socfpgaclk
->width
- 1, 0);
33 div
= ((readl(socfpgaclk
->hw
.reg
) & 0x7ff) + 1);
36 return parent_rate
/ div
;
39 static u8
clk_periclk_get_parent(struct clk_hw
*hwclk
)
41 struct socfpga_periph_clk
*socfpgaclk
= to_socfpga_periph_clk(hwclk
);
43 const char *name
= clk_hw_get_name(hwclk
);
45 clk_src
= readl(socfpgaclk
->hw
.reg
);
46 if (streq(name
, SOCFPGA_MPU_FREE_CLK
) ||
47 streq(name
, SOCFPGA_NOC_FREE_CLK
) ||
48 streq(name
, SOCFPGA_SDMMC_FREE_CLK
))
49 return (clk_src
>> CLK_MGR_FREE_SHIFT
) &
55 static const struct clk_ops periclk_ops
= {
56 .recalc_rate
= clk_periclk_recalc_rate
,
57 .get_parent
= clk_periclk_get_parent
,
60 static void __init
__socfpga_periph_init(struct device_node
*node
,
61 const struct clk_ops
*ops
)
64 struct clk_hw
*hw_clk
;
65 struct socfpga_periph_clk
*periph_clk
;
66 const char *clk_name
= node
->name
;
67 const char *parent_name
[SOCFPGA_MAX_PARENTS
];
68 struct clk_init_data init
;
73 of_property_read_u32(node
, "reg", ®
);
75 periph_clk
= kzalloc(sizeof(*periph_clk
), GFP_KERNEL
);
76 if (WARN_ON(!periph_clk
))
79 periph_clk
->hw
.reg
= clk_mgr_a10_base_addr
+ reg
;
81 rc
= of_property_read_u32_array(node
, "div-reg", div_reg
, 3);
83 periph_clk
->div_reg
= clk_mgr_a10_base_addr
+ div_reg
[0];
84 periph_clk
->shift
= div_reg
[1];
85 periph_clk
->width
= div_reg
[2];
87 periph_clk
->div_reg
= NULL
;
90 rc
= of_property_read_u32(node
, "fixed-divider", &fixed_div
);
92 periph_clk
->fixed_div
= 0;
94 periph_clk
->fixed_div
= fixed_div
;
96 of_property_read_string(node
, "clock-output-names", &clk_name
);
102 init
.num_parents
= of_clk_parent_fill(node
, parent_name
, SOCFPGA_MAX_PARENTS
);
103 init
.parent_names
= parent_name
;
105 periph_clk
->hw
.hw
.init
= &init
;
107 hw_clk
= &periph_clk
->hw
.hw
;
109 rc
= clk_hw_register(NULL
, hw_clk
);
111 pr_err("Could not register clock:%s\n", clk_name
);
112 goto err_clk_hw_register
;
115 rc
= of_clk_add_hw_provider(node
, of_clk_hw_simple_get
, hw_clk
);
117 pr_err("Could not register clock provider for node:%s\n",
119 goto err_of_clk_add_hw_provider
;
124 err_of_clk_add_hw_provider
:
125 clk_hw_unregister(hw_clk
);
130 void __init
socfpga_a10_periph_init(struct device_node
*node
)
132 __socfpga_periph_init(node
, &periclk_ops
);