blob: 1d779ad121692d88ecbcb84d4d579ece7515847e [file] [log] [blame]
Ulf Hansson3b01f872012-08-27 15:45:50 +02001/*
2 * PRCMU clock implementation for ux500 platform.
3 *
4 * Copyright (C) 2012 ST-Ericsson SA
5 * Author: Ulf Hansson <ulf.hansson@linaro.org>
6 *
7 * License terms: GNU General Public License (GPL) version 2
8 */
9
10#include <linux/clk-provider.h>
11#include <linux/clk-private.h>
12#include <linux/mfd/dbx500-prcmu.h>
13#include <linux/slab.h>
14#include <linux/io.h>
15#include <linux/err.h>
16#include "clk.h"
17
18#define to_clk_prcmu(_hw) container_of(_hw, struct clk_prcmu, hw)
19
20struct clk_prcmu {
21 struct clk_hw hw;
22 u8 cg_sel;
23 int is_enabled;
24};
25
26/* PRCMU clock operations. */
27
28static int clk_prcmu_prepare(struct clk_hw *hw)
29{
30 struct clk_prcmu *clk = to_clk_prcmu(hw);
31 return prcmu_request_clock(clk->cg_sel, true);
32}
33
34static void clk_prcmu_unprepare(struct clk_hw *hw)
35{
36 struct clk_prcmu *clk = to_clk_prcmu(hw);
37 if (prcmu_request_clock(clk->cg_sel, false))
38 pr_err("clk_prcmu: %s failed to disable %s.\n", __func__,
39 hw->init->name);
40}
41
42static int clk_prcmu_enable(struct clk_hw *hw)
43{
44 struct clk_prcmu *clk = to_clk_prcmu(hw);
45 clk->is_enabled = 1;
46 return 0;
47}
48
49static void clk_prcmu_disable(struct clk_hw *hw)
50{
51 struct clk_prcmu *clk = to_clk_prcmu(hw);
52 clk->is_enabled = 0;
53}
54
55static int clk_prcmu_is_enabled(struct clk_hw *hw)
56{
57 struct clk_prcmu *clk = to_clk_prcmu(hw);
58 return clk->is_enabled;
59}
60
61static unsigned long clk_prcmu_recalc_rate(struct clk_hw *hw,
62 unsigned long parent_rate)
63{
64 struct clk_prcmu *clk = to_clk_prcmu(hw);
65 return prcmu_clock_rate(clk->cg_sel);
66}
67
68static long clk_prcmu_round_rate(struct clk_hw *hw, unsigned long rate,
69 unsigned long *parent_rate)
70{
71 struct clk_prcmu *clk = to_clk_prcmu(hw);
72 return prcmu_round_clock_rate(clk->cg_sel, rate);
73}
74
75static int clk_prcmu_set_rate(struct clk_hw *hw, unsigned long rate,
76 unsigned long parent_rate)
77{
78 struct clk_prcmu *clk = to_clk_prcmu(hw);
79 return prcmu_set_clock_rate(clk->cg_sel, rate);
80}
81
82static int request_ape_opp100(bool enable)
83{
84 static int reqs;
85 int err = 0;
86
87 if (enable) {
88 if (!reqs)
89 err = prcmu_qos_add_requirement(PRCMU_QOS_APE_OPP,
90 "clock", 100);
91 if (!err)
92 reqs++;
93 } else {
94 reqs--;
95 if (!reqs)
96 prcmu_qos_remove_requirement(PRCMU_QOS_APE_OPP,
97 "clock");
98 }
99 return err;
100}
101
102static int clk_prcmu_opp_prepare(struct clk_hw *hw)
103{
104 int err;
105 struct clk_prcmu *clk = to_clk_prcmu(hw);
106
107 err = request_ape_opp100(true);
108 if (err) {
109 pr_err("clk_prcmu: %s failed to request APE OPP100 for %s.\n",
110 __func__, hw->init->name);
111 return err;
112 }
113
114 err = prcmu_request_clock(clk->cg_sel, true);
115 if (err)
116 request_ape_opp100(false);
117
118 return err;
119}
120
121static void clk_prcmu_opp_unprepare(struct clk_hw *hw)
122{
123 struct clk_prcmu *clk = to_clk_prcmu(hw);
124
125 if (prcmu_request_clock(clk->cg_sel, false))
126 goto out_error;
127 if (request_ape_opp100(false))
128 goto out_error;
129 return;
130
131out_error:
132 pr_err("clk_prcmu: %s failed to disable %s.\n", __func__,
133 hw->init->name);
134}
135
136static struct clk_ops clk_prcmu_scalable_ops = {
137 .prepare = clk_prcmu_prepare,
138 .unprepare = clk_prcmu_unprepare,
139 .enable = clk_prcmu_enable,
140 .disable = clk_prcmu_disable,
141 .is_enabled = clk_prcmu_is_enabled,
142 .recalc_rate = clk_prcmu_recalc_rate,
143 .round_rate = clk_prcmu_round_rate,
144 .set_rate = clk_prcmu_set_rate,
145};
146
147static struct clk_ops clk_prcmu_gate_ops = {
148 .prepare = clk_prcmu_prepare,
149 .unprepare = clk_prcmu_unprepare,
150 .enable = clk_prcmu_enable,
151 .disable = clk_prcmu_disable,
152 .is_enabled = clk_prcmu_is_enabled,
153 .recalc_rate = clk_prcmu_recalc_rate,
154};
155
156static struct clk_ops clk_prcmu_opp_gate_ops = {
157 .prepare = clk_prcmu_opp_prepare,
158 .unprepare = clk_prcmu_opp_unprepare,
159 .enable = clk_prcmu_enable,
160 .disable = clk_prcmu_disable,
161 .is_enabled = clk_prcmu_is_enabled,
162 .recalc_rate = clk_prcmu_recalc_rate,
163};
164
165static struct clk *clk_reg_prcmu(const char *name,
166 const char *parent_name,
167 u8 cg_sel,
168 unsigned long rate,
169 unsigned long flags,
170 struct clk_ops *clk_prcmu_ops)
171{
172 struct clk_prcmu *clk;
173 struct clk_init_data clk_prcmu_init;
174 struct clk *clk_reg;
175
176 if (!name) {
177 pr_err("clk_prcmu: %s invalid arguments passed\n", __func__);
178 return ERR_PTR(-EINVAL);
179 }
180
181 clk = kzalloc(sizeof(struct clk_prcmu), GFP_KERNEL);
182 if (!clk) {
183 pr_err("clk_prcmu: %s could not allocate clk\n", __func__);
184 return ERR_PTR(-ENOMEM);
185 }
186
187 clk->cg_sel = cg_sel;
188 clk->is_enabled = 1;
189 /* "rate" can be used for changing the initial frequency */
190 if (rate)
191 prcmu_set_clock_rate(cg_sel, rate);
192
193 clk_prcmu_init.name = name;
194 clk_prcmu_init.ops = clk_prcmu_ops;
195 clk_prcmu_init.flags = flags;
196 clk_prcmu_init.parent_names = (parent_name ? &parent_name : NULL);
197 clk_prcmu_init.num_parents = (parent_name ? 1 : 0);
198 clk->hw.init = &clk_prcmu_init;
199
200 clk_reg = clk_register(NULL, &clk->hw);
201 if (IS_ERR_OR_NULL(clk_reg))
202 goto free_clk;
203
204 return clk_reg;
205
206free_clk:
207 kfree(clk);
208 pr_err("clk_prcmu: %s failed to register clk\n", __func__);
209 return ERR_PTR(-ENOMEM);
210}
211
212struct clk *clk_reg_prcmu_scalable(const char *name,
213 const char *parent_name,
214 u8 cg_sel,
215 unsigned long rate,
216 unsigned long flags)
217{
218 return clk_reg_prcmu(name, parent_name, cg_sel, rate, flags,
219 &clk_prcmu_scalable_ops);
220}
221
222struct clk *clk_reg_prcmu_gate(const char *name,
223 const char *parent_name,
224 u8 cg_sel,
225 unsigned long flags)
226{
227 return clk_reg_prcmu(name, parent_name, cg_sel, 0, flags,
228 &clk_prcmu_gate_ops);
229}
230
231struct clk *clk_reg_prcmu_opp_gate(const char *name,
232 const char *parent_name,
233 u8 cg_sel,
234 unsigned long flags)
235{
236 return clk_reg_prcmu(name, parent_name, cg_sel, 0, flags,
237 &clk_prcmu_opp_gate_ops);
238}