blob: f78e5fa3f08de53fe2a497006999fa55d15d16a5 [file] [log] [blame]
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +05301// SPDX-License-Identifier: GPL-2.0+
2/*
Amit Singh Tomar8b520ac2020-04-19 19:28:30 +05303 * Common clock driver for Actions Semi SoCs.
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +05304 *
5 * Copyright (C) 2015 Actions Semi Co., Ltd.
6 * Copyright (C) 2018 Manivannan Sadhasivam <manivannan.sadhasivam@linaro.org>
7 */
8
9#include <common.h>
10#include <dm.h>
Amit Singh Tomar8b520ac2020-04-19 19:28:30 +053011#include "clk_owl.h"
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +053012#include <asm/io.h>
Amit Singh Tomar8b520ac2020-04-19 19:28:30 +053013#if defined(CONFIG_MACH_S900)
14#include <asm/arch-owl/regs_s900.h>
Amit Singh Tomar4939bee2020-04-19 19:28:28 +053015#include <dt-bindings/clock/actions,s900-cmu.h>
Amit Singh Tomar8b520ac2020-04-19 19:28:30 +053016#elif defined(CONFIG_MACH_S700)
17#include <asm/arch-owl/regs_s700.h>
18#include <dt-bindings/clock/actions,s700-cmu.h>
19#endif
Simon Glasscd93d622020-05-10 11:40:13 -060020#include <linux/bitops.h>
Simon Glassc05ed002020-05-10 11:40:11 -060021#include <linux/delay.h>
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +053022
23void owl_clk_init(struct owl_clk_priv *priv)
24{
25 u32 bus_clk = 0, core_pll, dev_pll;
26
Amit Singh Tomar8b520ac2020-04-19 19:28:30 +053027#if defined(CONFIG_MACH_S900)
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +053028 /* Enable ASSIST_PLL */
29 setbits_le32(priv->base + CMU_ASSISTPLL, BIT(0));
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +053030 udelay(PLL_STABILITY_WAIT_US);
Amit Singh Tomar8b520ac2020-04-19 19:28:30 +053031#endif
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +053032
33 /* Source HOSC to DEV_CLK */
34 clrbits_le32(priv->base + CMU_DEVPLL, CMU_DEVPLL_CLK);
35
36 /* Configure BUS_CLK */
37 bus_clk |= (CMU_PDBGDIV_DIV | CMU_PERDIV_DIV | CMU_NOCDIV_DIV |
38 CMU_DMMCLK_SRC | CMU_APBCLK_DIV | CMU_AHBCLK_DIV |
39 CMU_NOCCLK_SRC | CMU_CORECLK_HOSC);
40 writel(bus_clk, priv->base + CMU_BUSCLK);
41
42 udelay(PLL_STABILITY_WAIT_US);
43
44 /* Configure CORE_PLL */
45 core_pll = readl(priv->base + CMU_COREPLL);
46 core_pll |= (CMU_COREPLL_EN | CMU_COREPLL_HOSC_EN | CMU_COREPLL_OUT);
47 writel(core_pll, priv->base + CMU_COREPLL);
48
49 udelay(PLL_STABILITY_WAIT_US);
50
51 /* Configure DEV_PLL */
52 dev_pll = readl(priv->base + CMU_DEVPLL);
53 dev_pll |= (CMU_DEVPLL_EN | CMU_DEVPLL_OUT);
54 writel(dev_pll, priv->base + CMU_DEVPLL);
55
56 udelay(PLL_STABILITY_WAIT_US);
57
58 /* Source CORE_PLL for CORE_CLK */
59 clrsetbits_le32(priv->base + CMU_BUSCLK, CMU_CORECLK_MASK,
60 CMU_CORECLK_CPLL);
61
62 /* Source DEV_PLL for DEV_CLK */
63 setbits_le32(priv->base + CMU_DEVPLL, CMU_DEVPLL_CLK);
64
65 udelay(PLL_STABILITY_WAIT_US);
66}
67
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +053068int owl_clk_enable(struct clk *clk)
69{
70 struct owl_clk_priv *priv = dev_get_priv(clk->dev);
Amit Singh Tomar8b520ac2020-04-19 19:28:30 +053071 enum owl_soc model = dev_get_driver_data(clk->dev);
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +053072
73 switch (clk->id) {
Amit Singh Tomar4939bee2020-04-19 19:28:28 +053074 case CLK_UART5:
Amit Singh Tomar8b520ac2020-04-19 19:28:30 +053075 if (model != S900)
76 return -EINVAL;
77 /* Source HOSC for UART5 interface */
78 clrbits_le32(priv->base + CMU_UART5CLK, CMU_UARTCLK_SRC_DEVPLL);
79 /* Enable UART5 interface clock */
80 setbits_le32(priv->base + CMU_DEVCLKEN1, CMU_DEVCLKEN1_UART5);
81 break;
82 case CLK_UART3:
83 if (model != S700)
84 return -EINVAL;
85 /* Source HOSC for UART3 interface */
86 clrbits_le32(priv->base + CMU_UART3CLK, CMU_UARTCLK_SRC_DEVPLL);
87 /* Enable UART3 interface clock */
88 setbits_le32(priv->base + CMU_DEVCLKEN1, CMU_DEVCLKEN1_UART3);
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +053089 break;
Amit Singh Tomar3a217342020-05-09 19:55:09 +053090 case CLK_RMII_REF:
91 case CLK_ETHERNET:
92 setbits_le32(priv->base + CMU_DEVCLKEN1, CMU_DEVCLKEN1_ETH);
93 setbits_le32(priv->base + CMU_ETHERNETPLL, 5);
94 break;
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +053095 default:
Amit Singh Tomar8b520ac2020-04-19 19:28:30 +053096 return -EINVAL;
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +053097 }
98
99 return 0;
100}
101
102int owl_clk_disable(struct clk *clk)
103{
104 struct owl_clk_priv *priv = dev_get_priv(clk->dev);
Amit Singh Tomar8b520ac2020-04-19 19:28:30 +0530105 enum owl_soc model = dev_get_driver_data(clk->dev);
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +0530106
107 switch (clk->id) {
Amit Singh Tomar4939bee2020-04-19 19:28:28 +0530108 case CLK_UART5:
Amit Singh Tomar8b520ac2020-04-19 19:28:30 +0530109 if (model != S900)
110 return -EINVAL;
111 /* Disable UART5 interface clock */
112 clrbits_le32(priv->base + CMU_DEVCLKEN1, CMU_DEVCLKEN1_UART5);
113 break;
114 case CLK_UART3:
115 if (model != S700)
116 return -EINVAL;
117 /* Disable UART3 interface clock */
118 clrbits_le32(priv->base + CMU_DEVCLKEN1, CMU_DEVCLKEN1_UART3);
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +0530119 break;
Amit Singh Tomar3a217342020-05-09 19:55:09 +0530120 case CLK_RMII_REF:
121 case CLK_ETHERNET:
122 clrbits_le32(priv->base + CMU_DEVCLKEN1, CMU_DEVCLKEN1_ETH);
123 break;
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +0530124 default:
Amit Singh Tomar8b520ac2020-04-19 19:28:30 +0530125 return -EINVAL;
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +0530126 }
127
128 return 0;
129}
130
Amit Singh Tomar234c1672021-11-28 17:02:20 +0530131static ulong owl_clk_get_rate(struct clk *clk)
132{
133 ulong rate;
134
135 switch (clk->id) {
136 default:
137 return -ENOENT;
138 }
139
140 return rate;
141}
142
143static ulong owl_clk_set_rate(struct clk *clk, ulong rate)
144{
145 ulong new_rate;
146
147 switch (clk->id) {
148 default:
149 return -ENOENT;
150 }
151
152 return new_rate;
153}
154
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +0530155static int owl_clk_probe(struct udevice *dev)
156{
157 struct owl_clk_priv *priv = dev_get_priv(dev);
158
159 priv->base = dev_read_addr(dev);
160 if (priv->base == FDT_ADDR_T_NONE)
161 return -EINVAL;
162
163 /* setup necessary clocks */
164 owl_clk_init(priv);
165
166 return 0;
167}
168
Amit Singh Tomar8b520ac2020-04-19 19:28:30 +0530169static const struct clk_ops owl_clk_ops = {
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +0530170 .enable = owl_clk_enable,
171 .disable = owl_clk_disable,
Amit Singh Tomar234c1672021-11-28 17:02:20 +0530172 .get_rate = owl_clk_get_rate,
173 .set_rate = owl_clk_set_rate,
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +0530174};
175
176static const struct udevice_id owl_clk_ids[] = {
Amit Singh Tomar8b520ac2020-04-19 19:28:30 +0530177#if defined(CONFIG_MACH_S900)
178 { .compatible = "actions,s900-cmu", .data = S900 },
179#elif defined(CONFIG_MACH_S700)
180 { .compatible = "actions,s700-cmu", .data = S700 },
181#endif
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +0530182 { }
183};
184
185U_BOOT_DRIVER(clk_owl) = {
Amit Singh Tomar8b520ac2020-04-19 19:28:30 +0530186 .name = "clk_owl",
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +0530187 .id = UCLASS_CLK,
188 .of_match = owl_clk_ids,
189 .ops = &owl_clk_ops,
Simon Glass41575d82020-12-03 16:55:17 -0700190 .priv_auto = sizeof(struct owl_clk_priv),
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +0530191 .probe = owl_clk_probe,
Manivannan Sadhasivamae485b52018-06-14 23:38:35 +0530192};