Linux kernel mirror (for testing)
git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel
os
linux
1// SPDX-License-Identifier: GPL-2.0
2#include <linux/clk.h>
3#include <linux/device.h>
4#include <linux/export.h>
5#include <linux/gfp.h>
6
7struct devm_clk_state {
8 struct clk *clk;
9 void (*exit)(struct clk *clk);
10};
11
12static void devm_clk_release(struct device *dev, void *res)
13{
14 struct devm_clk_state *state = res;
15
16 if (state->exit)
17 state->exit(state->clk);
18
19 clk_put(state->clk);
20}
21
22static struct clk *__devm_clk_get(struct device *dev, const char *id,
23 struct clk *(*get)(struct device *dev, const char *id),
24 int (*init)(struct clk *clk),
25 void (*exit)(struct clk *clk))
26{
27 struct devm_clk_state *state;
28 struct clk *clk;
29 int ret;
30
31 state = devres_alloc(devm_clk_release, sizeof(*state), GFP_KERNEL);
32 if (!state)
33 return ERR_PTR(-ENOMEM);
34
35 clk = get(dev, id);
36 if (IS_ERR(clk)) {
37 ret = PTR_ERR(clk);
38 goto err_clk_get;
39 }
40
41 if (init) {
42 ret = init(clk);
43 if (ret)
44 goto err_clk_init;
45 }
46
47 state->clk = clk;
48 state->exit = exit;
49
50 devres_add(dev, state);
51
52 return clk;
53
54err_clk_init:
55
56 clk_put(clk);
57err_clk_get:
58
59 devres_free(state);
60 return ERR_PTR(ret);
61}
62
63struct clk *devm_clk_get(struct device *dev, const char *id)
64{
65 return __devm_clk_get(dev, id, clk_get, NULL, NULL);
66}
67EXPORT_SYMBOL(devm_clk_get);
68
69struct clk *devm_clk_get_prepared(struct device *dev, const char *id)
70{
71 return __devm_clk_get(dev, id, clk_get, clk_prepare, clk_unprepare);
72}
73EXPORT_SYMBOL_GPL(devm_clk_get_prepared);
74
75struct clk *devm_clk_get_enabled(struct device *dev, const char *id)
76{
77 return __devm_clk_get(dev, id, clk_get,
78 clk_prepare_enable, clk_disable_unprepare);
79}
80EXPORT_SYMBOL_GPL(devm_clk_get_enabled);
81
82struct clk *devm_clk_get_optional(struct device *dev, const char *id)
83{
84 return __devm_clk_get(dev, id, clk_get_optional, NULL, NULL);
85}
86EXPORT_SYMBOL(devm_clk_get_optional);
87
88struct clk *devm_clk_get_optional_prepared(struct device *dev, const char *id)
89{
90 return __devm_clk_get(dev, id, clk_get_optional,
91 clk_prepare, clk_unprepare);
92}
93EXPORT_SYMBOL_GPL(devm_clk_get_optional_prepared);
94
95struct clk *devm_clk_get_optional_enabled(struct device *dev, const char *id)
96{
97 return __devm_clk_get(dev, id, clk_get_optional,
98 clk_prepare_enable, clk_disable_unprepare);
99}
100EXPORT_SYMBOL_GPL(devm_clk_get_optional_enabled);
101
102struct clk *devm_clk_get_optional_enabled_with_rate(struct device *dev,
103 const char *id,
104 unsigned long rate)
105{
106 struct clk *clk;
107 int ret;
108
109 clk = __devm_clk_get(dev, id, clk_get_optional, NULL,
110 clk_disable_unprepare);
111 if (IS_ERR(clk))
112 return ERR_CAST(clk);
113
114 ret = clk_set_rate(clk, rate);
115 if (ret)
116 goto out_put_clk;
117
118 ret = clk_prepare_enable(clk);
119 if (ret)
120 goto out_put_clk;
121
122 return clk;
123
124out_put_clk:
125 devm_clk_put(dev, clk);
126 return ERR_PTR(ret);
127}
128EXPORT_SYMBOL_GPL(devm_clk_get_optional_enabled_with_rate);
129
130struct clk_bulk_devres {
131 struct clk_bulk_data *clks;
132 int num_clks;
133};
134
135static void devm_clk_bulk_release(struct device *dev, void *res)
136{
137 struct clk_bulk_devres *devres = res;
138
139 clk_bulk_put(devres->num_clks, devres->clks);
140}
141
142static int __devm_clk_bulk_get(struct device *dev, int num_clks,
143 struct clk_bulk_data *clks, bool optional)
144{
145 struct clk_bulk_devres *devres;
146 int ret;
147
148 devres = devres_alloc(devm_clk_bulk_release,
149 sizeof(*devres), GFP_KERNEL);
150 if (!devres)
151 return -ENOMEM;
152
153 if (optional)
154 ret = clk_bulk_get_optional(dev, num_clks, clks);
155 else
156 ret = clk_bulk_get(dev, num_clks, clks);
157 if (!ret) {
158 devres->clks = clks;
159 devres->num_clks = num_clks;
160 devres_add(dev, devres);
161 } else {
162 devres_free(devres);
163 }
164
165 return ret;
166}
167
168int __must_check devm_clk_bulk_get(struct device *dev, int num_clks,
169 struct clk_bulk_data *clks)
170{
171 return __devm_clk_bulk_get(dev, num_clks, clks, false);
172}
173EXPORT_SYMBOL_GPL(devm_clk_bulk_get);
174
175int __must_check devm_clk_bulk_get_optional(struct device *dev, int num_clks,
176 struct clk_bulk_data *clks)
177{
178 return __devm_clk_bulk_get(dev, num_clks, clks, true);
179}
180EXPORT_SYMBOL_GPL(devm_clk_bulk_get_optional);
181
182static void devm_clk_bulk_release_enable(struct device *dev, void *res)
183{
184 struct clk_bulk_devres *devres = res;
185
186 clk_bulk_disable_unprepare(devres->num_clks, devres->clks);
187 clk_bulk_put(devres->num_clks, devres->clks);
188}
189
190static int __devm_clk_bulk_get_enable(struct device *dev, int num_clks,
191 struct clk_bulk_data *clks, bool optional)
192{
193 struct clk_bulk_devres *devres;
194 int ret;
195
196 devres = devres_alloc(devm_clk_bulk_release_enable,
197 sizeof(*devres), GFP_KERNEL);
198 if (!devres)
199 return -ENOMEM;
200
201 if (optional)
202 ret = clk_bulk_get_optional(dev, num_clks, clks);
203 else
204 ret = clk_bulk_get(dev, num_clks, clks);
205 if (ret)
206 goto err_clk_get;
207
208 ret = clk_bulk_prepare_enable(num_clks, clks);
209 if (ret)
210 goto err_clk_prepare;
211
212 devres->clks = clks;
213 devres->num_clks = num_clks;
214 devres_add(dev, devres);
215
216 return 0;
217
218err_clk_prepare:
219 clk_bulk_put(num_clks, clks);
220err_clk_get:
221 devres_free(devres);
222 return ret;
223}
224
225int __must_check devm_clk_bulk_get_optional_enable(struct device *dev, int num_clks,
226 struct clk_bulk_data *clks)
227{
228 return __devm_clk_bulk_get_enable(dev, num_clks, clks, true);
229}
230EXPORT_SYMBOL_GPL(devm_clk_bulk_get_optional_enable);
231
232static void devm_clk_bulk_release_all(struct device *dev, void *res)
233{
234 struct clk_bulk_devres *devres = res;
235
236 clk_bulk_put_all(devres->num_clks, devres->clks);
237}
238
239int __must_check devm_clk_bulk_get_all(struct device *dev,
240 struct clk_bulk_data **clks)
241{
242 struct clk_bulk_devres *devres;
243 int ret;
244
245 devres = devres_alloc(devm_clk_bulk_release_all,
246 sizeof(*devres), GFP_KERNEL);
247 if (!devres)
248 return -ENOMEM;
249
250 ret = clk_bulk_get_all(dev, &devres->clks);
251 if (ret > 0) {
252 *clks = devres->clks;
253 devres->num_clks = ret;
254 devres_add(dev, devres);
255 } else {
256 devres_free(devres);
257 }
258
259 return ret;
260}
261EXPORT_SYMBOL_GPL(devm_clk_bulk_get_all);
262
263static void devm_clk_bulk_release_all_enable(struct device *dev, void *res)
264{
265 struct clk_bulk_devres *devres = res;
266
267 clk_bulk_disable_unprepare(devres->num_clks, devres->clks);
268 clk_bulk_put_all(devres->num_clks, devres->clks);
269}
270
271int __must_check devm_clk_bulk_get_all_enabled(struct device *dev,
272 struct clk_bulk_data **clks)
273{
274 struct clk_bulk_devres *devres;
275 int ret;
276
277 devres = devres_alloc(devm_clk_bulk_release_all_enable,
278 sizeof(*devres), GFP_KERNEL);
279 if (!devres)
280 return -ENOMEM;
281
282 ret = clk_bulk_get_all(dev, &devres->clks);
283 if (ret > 0) {
284 *clks = devres->clks;
285 devres->num_clks = ret;
286 } else {
287 devres_free(devres);
288 return ret;
289 }
290
291 ret = clk_bulk_prepare_enable(devres->num_clks, *clks);
292 if (!ret) {
293 devres_add(dev, devres);
294 } else {
295 clk_bulk_put_all(devres->num_clks, devres->clks);
296 devres_free(devres);
297 return ret;
298 }
299
300 return devres->num_clks;
301}
302EXPORT_SYMBOL_GPL(devm_clk_bulk_get_all_enabled);
303
304static int devm_clk_match(struct device *dev, void *res, void *data)
305{
306 struct clk **c = res;
307 if (!c || !*c) {
308 WARN_ON(!c || !*c);
309 return 0;
310 }
311 return *c == data;
312}
313
314void devm_clk_put(struct device *dev, struct clk *clk)
315{
316 int ret;
317
318 ret = devres_release(dev, devm_clk_release, devm_clk_match, clk);
319
320 WARN_ON(ret);
321}
322EXPORT_SYMBOL(devm_clk_put);
323
324struct clk *devm_get_clk_from_child(struct device *dev,
325 struct device_node *np, const char *con_id)
326{
327 struct devm_clk_state *state;
328 struct clk *clk;
329
330 state = devres_alloc(devm_clk_release, sizeof(*state), GFP_KERNEL);
331 if (!state)
332 return ERR_PTR(-ENOMEM);
333
334 clk = of_clk_get_by_name(np, con_id);
335 if (!IS_ERR(clk)) {
336 state->clk = clk;
337 devres_add(dev, state);
338 } else {
339 devres_free(state);
340 }
341
342 return clk;
343}
344EXPORT_SYMBOL(devm_get_clk_from_child);