blob: d0f5bb375300a7b73d0d7760fa8d517675000a83 [file] [log] [blame]
Igor Prusov1fcf1902023-09-25 18:52:09 +03001// SPDX-License-Identifier: GPL-2.0+
2/*
3 * (C) Copyright 2023 SberDevices, Inc.
4 * Author: Igor Prusov <ivprusov@salutedevices.com>
5 */
6
7#include <common.h>
8#include <clk-uclass.h>
9#include <dm.h>
10#include <regmap.h>
11#include <asm/arch/clock-a1.h>
12#include <dt-bindings/clock/amlogic,a1-pll-clkc.h>
13#include <dt-bindings/clock/amlogic,a1-peripherals-clkc.h>
14#include "clk_meson.h"
15
16/*
17 * This driver supports both PLL and peripherals clock sources.
18 * Following operations are supported:
19 * - calculating clock frequency on a limited tree
20 * - reading muxes and dividers
21 * - enabling/disabling gates without propagation
22 * - reparenting without rate propagation, only on muxes
23 * - setting rates with limited reparenting, only on dividers with mux parent
24 */
25
26#define NR_CLKS 154
27#define NR_PLL_CLKS 11
28
29/* External clock IDs. Those should not overlap with regular IDs */
30#define EXTERNAL_XTAL (NR_CLKS + 0)
31#define EXTERNAL_FCLK_DIV2 (NR_CLKS + 1)
32#define EXTERNAL_FCLK_DIV3 (NR_CLKS + 2)
33#define EXTERNAL_FCLK_DIV5 (NR_CLKS + 3)
34#define EXTERNAL_FCLK_DIV7 (NR_CLKS + 4)
35
36#define EXTERNAL_FIXPLL_IN (NR_PLL_CLKS + 1)
37
38#define SET_PARM_VALUE(_priv, _parm, _val) \
39 regmap_update_bits((_priv)->map, (_parm)->reg_off, \
40 SETPMASK((_parm)->width, (_parm)->shift), \
41 (_val) << (_parm)->shift)
42
43#define GET_PARM_VALUE(_priv, _parm) \
44({ \
45 uint _reg; \
46 regmap_read((_priv)->map, (_parm)->reg_off, &_reg); \
47 PARM_GET((_parm)->width, (_parm)->shift, _reg); \
48})
49
50struct meson_clk {
51 struct regmap *map;
52};
53
54/**
55 * enum meson_clk_type - The type of clock
56 * @MESON_CLK_ANY: Special value that matches any clock type
57 * @MESON_CLK_GATE: This clock is a gate
58 * @MESON_CLK_MUX: This clock is a multiplexer
59 * @MESON_CLK_DIV: This clock is a configurable divider
60 * @MESON_CLK_FIXED_DIV: This clock is a configurable divider
61 * @MESON_CLK_EXTERNAL: This is an external clock from different clock provider
62 * @MESON_CLK_PLL: This is a PLL
63 */
64enum meson_clk_type {
65 MESON_CLK_ANY = 0,
66 MESON_CLK_GATE,
67 MESON_CLK_MUX,
68 MESON_CLK_DIV,
69 MESON_CLK_FIXED_DIV,
70 MESON_CLK_EXTERNAL,
71 MESON_CLK_PLL,
72};
73
74/**
75 * struct meson_clk_info - The parameters defining a clock
76 * @name: Name of the clock
77 * @parm: Register bits description for muxes and dividers
78 * @div: Fixed divider value
79 * @parents: List of parent clock IDs
80 * @type: Clock type
81 */
82struct meson_clk_info {
83 const char *name;
84 union {
85 const struct parm *parm;
86 u8 div;
87 };
88 const unsigned int *parents;
89 const enum meson_clk_type type;
90};
91
92/**
93 * struct meson_clk_data - Clocks supported by clock provider
94 * @num_clocks: Number of clocks
95 * @clocks: Array of clock descriptions
96 *
97 */
98struct meson_clk_data {
99 const u8 num_clocks;
100 const struct meson_clk_info **clocks;
101};
102
103/* Clock description initialization macros */
104
105/* A multiplexer */
106#define CLK_MUX(_name, _reg, _shift, _width, ...) \
107 (&(struct meson_clk_info){ \
108 .parents = (const unsigned int[])__VA_ARGS__, \
109 .parm = &(struct parm) { \
110 .reg_off = (_reg), \
111 .shift = (_shift), \
112 .width = (_width), \
113 }, \
114 .name = (_name), \
115 .type = MESON_CLK_MUX, \
116 })
117
118/* A divider with an integral divisor */
119#define CLK_DIV(_name, _reg, _shift, _width, _parent) \
120 (&(struct meson_clk_info){ \
121 .parents = (const unsigned int[]) { (_parent) }, \
122 .parm = &(struct parm) { \
123 .reg_off = (_reg), \
124 .shift = (_shift), \
125 .width = (_width), \
126 }, \
127 .name = (_name), \
128 .type = MESON_CLK_DIV, \
129 })
130
131/* A fixed divider */
132#define CLK_DIV_FIXED(_name, _div, _parent) \
133 (&(struct meson_clk_info){ \
134 .parents = (const unsigned int[]) { (_parent) }, \
135 .div = (_div), \
136 .name = (_name), \
137 .type = MESON_CLK_FIXED_DIV, \
138 })
139
140/* An external clock */
141#define CLK_EXTERNAL(_name) \
142 (&(struct meson_clk_info){ \
143 .name = (_name), \
144 .parents = (const unsigned int[]) { -ENOENT }, \
145 .type = MESON_CLK_EXTERNAL, \
146 })
147
148/* A clock gate */
149#define CLK_GATE(_name, _reg, _shift, _parent) \
150 (&(struct meson_clk_info){ \
151 .parents = (const unsigned int[]) { (_parent) }, \
152 .parm = &(struct parm) { \
153 .reg_off = (_reg), \
154 .shift = (_shift), \
155 .width = 1, \
156 }, \
157 .name = (_name), \
158 .type = MESON_CLK_GATE, \
159 })
160
161/* A PLL clock */
162#define CLK_PLL(_name, _parent, ...) \
163 (&(struct meson_clk_info){ \
164 .name = (_name), \
165 .parents = (const unsigned int[]) { (_parent) }, \
166 .parm = (const struct parm[])__VA_ARGS__, \
167 .type = MESON_CLK_PLL, \
168 })
169
170/* A1 peripherals clocks */
171static const struct meson_clk_info *meson_clocks[] = {
172 [CLKID_SPIFC_SEL] = CLK_MUX("spifc_sel", A1_SPIFC_CLK_CTRL, 9, 2, {
173 EXTERNAL_FCLK_DIV2,
174 EXTERNAL_FCLK_DIV3,
175 EXTERNAL_FCLK_DIV5,
176 -ENOENT,
177 }),
178 [CLKID_SPIFC_SEL2] = CLK_MUX("spifc_sel2", A1_SPIFC_CLK_CTRL, 15, 1, {
179 CLKID_SPIFC_DIV,
180 EXTERNAL_XTAL,
181 }),
182 [CLKID_USB_BUS_SEL] = CLK_MUX("usb_bus_sel", A1_USB_BUSCLK_CTRL, 9, 2, {
183 -ENOENT,
184 CLKID_SYS,
185 EXTERNAL_FCLK_DIV3,
186 EXTERNAL_FCLK_DIV5,
187 }),
188 [CLKID_SYS] = CLK_MUX("sys", A1_SYS_CLK_CTRL0, 31, 1, {
189 CLKID_SYS_A,
190 CLKID_SYS_B,
191 }),
192 [CLKID_SYS_A_SEL] = CLK_MUX("sys_a_sel", A1_SYS_CLK_CTRL0, 10, 3, {
193 -ENOENT,
194 EXTERNAL_FCLK_DIV2,
195 EXTERNAL_FCLK_DIV3,
196 EXTERNAL_FCLK_DIV5,
197 -ENOENT,
198 -ENOENT,
199 -ENOENT,
200 -ENOENT,
201 }),
202 [CLKID_SYS_B_SEL] = CLK_MUX("sys_b_sel", A1_SYS_CLK_CTRL0, 26, 3, {
203 -ENOENT,
204 EXTERNAL_FCLK_DIV2,
205 EXTERNAL_FCLK_DIV3,
206 EXTERNAL_FCLK_DIV5,
207 -ENOENT,
208 -ENOENT,
209 -ENOENT,
210 -ENOENT,
211 }),
212
213 [CLKID_SPIFC_DIV] = CLK_DIV("spifc_div", A1_SPIFC_CLK_CTRL, 0, 8,
214 CLKID_SPIFC_SEL
215 ),
216 [CLKID_USB_BUS_DIV] = CLK_DIV("usb_bus_div", A1_USB_BUSCLK_CTRL, 0, 8,
217 CLKID_USB_BUS_SEL
218 ),
219 [CLKID_SYS_A_DIV] = CLK_DIV("sys_a_div", A1_SYS_CLK_CTRL0, 0, 10,
220 CLKID_SYS_A_SEL
221 ),
222 [CLKID_SYS_B_DIV] = CLK_DIV("sys_b_div", A1_SYS_CLK_CTRL0, 16, 10,
223 CLKID_SYS_B_SEL
224 ),
225
226 [CLKID_SPIFC] = CLK_GATE("spifc", A1_SPIFC_CLK_CTRL, 8,
227 CLKID_SPIFC_SEL2
228 ),
229 [CLKID_USB_BUS] = CLK_GATE("usb_bus", A1_USB_BUSCLK_CTRL, 8,
230 CLKID_USB_BUS_DIV
231 ),
232 [CLKID_SYS_A] = CLK_GATE("sys_a", A1_SYS_CLK_CTRL0, 13,
233 CLKID_SYS_A_DIV
234 ),
235 [CLKID_SYS_B] = CLK_GATE("sys_b", A1_SYS_CLK_CTRL0, 29,
236 CLKID_SYS_B_DIV
237 ),
238 [CLKID_FIXPLL_IN] = CLK_GATE("fixpll_in", A1_SYS_OSCIN_CTRL, 1,
239 EXTERNAL_XTAL
240 ),
Igor Prusov7db7cce2023-10-05 11:54:27 +0300241 [CLKID_USB_PHY_IN] = CLK_GATE("usb_phy_in", A1_SYS_OSCIN_CTRL, 2,
242 EXTERNAL_XTAL
243 ),
Alexey Romanov8ec79032023-11-01 17:04:56 +0300244 [CLKID_USB_CTRL_IN] = CLK_GATE("usb_ctrl_in", A1_SYS_OSCIN_CTRL, 3,
245 EXTERNAL_XTAL
246 ),
247 [CLKID_USB_CTRL] = CLK_GATE("usb_ctrl", A1_SYS_CLK_EN0, 28,
248 CLKID_SYS
249 ),
Igor Prusov7db7cce2023-10-05 11:54:27 +0300250 [CLKID_USB_PHY] = CLK_GATE("usb_phy", A1_SYS_CLK_EN0, 27,
251 CLKID_SYS
252 ),
Igor Prusov1fcf1902023-09-25 18:52:09 +0300253 [CLKID_SARADC] = CLK_GATE("saradc", A1_SAR_ADC_CLK_CTR, 8,
254 -ENOENT
255 ),
256 [CLKID_SARADC_EN] = CLK_GATE("saradc_en", A1_SYS_CLK_EN0, 13,
257 CLKID_SYS
258 ),
259
260 [EXTERNAL_XTAL] = CLK_EXTERNAL("xtal"),
261 [EXTERNAL_FCLK_DIV2] = CLK_EXTERNAL("fclk_div2"),
262 [EXTERNAL_FCLK_DIV3] = CLK_EXTERNAL("fclk_div3"),
263 [EXTERNAL_FCLK_DIV5] = CLK_EXTERNAL("fclk_div5"),
264 [EXTERNAL_FCLK_DIV7] = CLK_EXTERNAL("fclk_div7"),
265};
266
267/* A1 PLL clocks */
268static const struct meson_clk_info *meson_pll_clocks[] = {
269 [EXTERNAL_FIXPLL_IN] = CLK_EXTERNAL("fixpll_in"),
270
271 [CLKID_FIXED_PLL_DCO] = CLK_PLL("fixed_pll_dco", EXTERNAL_FIXPLL_IN, {
272 {A1_ANACTRL_FIXPLL_CTRL0, 0, 8},
273 {A1_ANACTRL_FIXPLL_CTRL0, 10, 5},
274 }),
275
276 [CLKID_FCLK_DIV2_DIV] = CLK_DIV_FIXED("fclk_div2_div", 2,
277 CLKID_FIXED_PLL
278 ),
279 [CLKID_FCLK_DIV3_DIV] = CLK_DIV_FIXED("fclk_div3_div", 3,
280 CLKID_FIXED_PLL
281 ),
282 [CLKID_FCLK_DIV5_DIV] = CLK_DIV_FIXED("fclk_div5_div", 5,
283 CLKID_FIXED_PLL
284 ),
285 [CLKID_FCLK_DIV7_DIV] = CLK_DIV_FIXED("fclk_div7_div", 7,
286 CLKID_FIXED_PLL
287 ),
288
289 [CLKID_FIXED_PLL] = CLK_GATE("fixed_pll", A1_ANACTRL_FIXPLL_CTRL0, 20,
290 CLKID_FIXED_PLL_DCO
291 ),
292 [CLKID_FCLK_DIV2] = CLK_GATE("fclk_div2", A1_ANACTRL_FIXPLL_CTRL0, 21,
293 CLKID_FCLK_DIV2_DIV
294 ),
295 [CLKID_FCLK_DIV3] = CLK_GATE("fclk_div3", A1_ANACTRL_FIXPLL_CTRL0, 22,
296 CLKID_FCLK_DIV3_DIV
297 ),
298 [CLKID_FCLK_DIV5] = CLK_GATE("fclk_div5", A1_ANACTRL_FIXPLL_CTRL0, 23,
299 CLKID_FCLK_DIV5_DIV
300 ),
301 [CLKID_FCLK_DIV7] = CLK_GATE("fclk_div7", A1_ANACTRL_FIXPLL_CTRL0, 24,
302 CLKID_FCLK_DIV7_DIV
303 ),
304};
305
306static const struct meson_clk_info *meson_clk_get_info(struct clk *clk, ulong id,
307 enum meson_clk_type type)
308{
309 struct meson_clk_data *data;
310 const struct meson_clk_info *info;
311
312 data = (struct meson_clk_data *)dev_get_driver_data(clk->dev);
313 if (id >= data->num_clocks)
314 return ERR_PTR(-EINVAL);
315
316 info = data->clocks[id];
317 if (!info)
318 return ERR_PTR(-ENOENT);
319
320 if (type != MESON_CLK_ANY && type != info->type)
321 return ERR_PTR(-EINVAL);
322
323 return info;
324}
325
326static ulong meson_clk_get_rate_by_id(struct clk *clk, unsigned long id);
327
328static int meson_set_gate(struct clk *clk, bool on)
329{
330 struct meson_clk *priv = dev_get_priv(clk->dev);
331 const struct meson_clk_info *info;
332
333 debug("%s: %sabling %lu\n", __func__, on ? "en" : "dis", clk->id);
334
335 info = meson_clk_get_info(clk, clk->id, MESON_CLK_ANY);
336 if (IS_ERR(info))
337 return PTR_ERR(info);
338
339 SET_PARM_VALUE(priv, info->parm, on);
340
341 return 0;
342}
343
344static int meson_clk_enable(struct clk *clk)
345{
346 return meson_set_gate(clk, true);
347}
348
349static int meson_clk_disable(struct clk *clk)
350{
351 return meson_set_gate(clk, false);
352}
353
354static ulong meson_div_get_rate(struct clk *clk, unsigned long id)
355{
356 struct meson_clk *priv = dev_get_priv(clk->dev);
357 u16 n;
358 ulong rate;
359 const struct meson_clk_info *info;
360
361 info = meson_clk_get_info(clk, id, MESON_CLK_DIV);
362 if (IS_ERR(info))
363 return PTR_ERR(info);
364
365 /* Actual divider value is (field value + 1), hence the increment */
366 n = GET_PARM_VALUE(priv, info->parm) + 1;
367
368 rate = meson_clk_get_rate_by_id(clk, info->parents[0]);
369
370 return rate / n;
371}
372
373static int meson_clk_get_parent(struct clk *clk, unsigned long id)
374{
375 uint reg = 0;
376 struct meson_clk *priv = dev_get_priv(clk->dev);
377 const struct meson_clk_info *info;
378
379 info = meson_clk_get_info(clk, id, MESON_CLK_ANY);
380 if (IS_ERR(info))
381 return PTR_ERR(info);
382
383 /* For muxes we read currently selected parent from register,
384 * for other types there is always only one element in parents array.
385 */
386 if (info->type == MESON_CLK_MUX) {
387 reg = GET_PARM_VALUE(priv, info->parm);
388 if (IS_ERR_VALUE(reg))
389 return reg;
390 }
391
392 return info->parents[reg];
393}
394
395static ulong meson_pll_get_rate(struct clk *clk, unsigned long id)
396{
397 struct meson_clk *priv = dev_get_priv(clk->dev);
398 const struct meson_clk_info *info;
399 const struct parm *pm, *pn;
400 ulong parent_rate_mhz;
401 unsigned int parent;
402 u16 n, m;
403
404 info = meson_clk_get_info(clk, id, MESON_CLK_ANY);
405 if (IS_ERR(info))
406 return PTR_ERR(info);
407
408 pm = &info->parm[0];
409 pn = &info->parm[1];
410
411 n = GET_PARM_VALUE(priv, pn);
412 m = GET_PARM_VALUE(priv, pm);
413
414 if (n == 0)
415 return -EINVAL;
416
417 parent = info->parents[0];
418 parent_rate_mhz = meson_clk_get_rate_by_id(clk, parent) / 1000000;
419
420 return parent_rate_mhz * m / n * 1000000;
421}
422
423static ulong meson_clk_get_rate_by_id(struct clk *clk, unsigned long id)
424{
425 ulong rate, parent;
426 const struct meson_clk_info *info;
427
428 if (IS_ERR_VALUE(id))
429 return id;
430
431 info = meson_clk_get_info(clk, id, MESON_CLK_ANY);
432 if (IS_ERR(info))
433 return PTR_ERR(info);
434
435 switch (info->type) {
436 case MESON_CLK_PLL:
437 rate = meson_pll_get_rate(clk, id);
438 break;
439 case MESON_CLK_GATE:
440 case MESON_CLK_MUX:
441 parent = meson_clk_get_parent(clk, id);
442 rate = meson_clk_get_rate_by_id(clk, parent);
443 break;
444 case MESON_CLK_DIV:
445 rate = meson_div_get_rate(clk, id);
446 break;
447 case MESON_CLK_FIXED_DIV:
448 parent = meson_clk_get_parent(clk, id);
449 rate = meson_clk_get_rate_by_id(clk, parent) / info->div;
450 break;
451 case MESON_CLK_EXTERNAL: {
452 int ret;
453 struct clk external_clk;
454
455 ret = clk_get_by_name(clk->dev, info->name, &external_clk);
456 if (ret)
457 return ret;
458
459 rate = clk_get_rate(&external_clk);
460 break;
461 }
462 default:
463 rate = -EINVAL;
464 break;
465 }
466
467 return rate;
468}
469
470static ulong meson_clk_get_rate(struct clk *clk)
471{
472 return meson_clk_get_rate_by_id(clk, clk->id);
473}
474
475/* This implements rate propagation for dividers placed after multiplexer:
476 * ---------|\
477 * ..... | |---DIV--
478 * ---------|/
479 */
480static ulong meson_composite_set_rate(struct clk *clk, ulong id, ulong rate)
481{
482 unsigned int i, best_div_val;
483 unsigned long best_delta, best_parent;
484 const struct meson_clk_info *div;
485 const struct meson_clk_info *mux;
486 struct meson_clk *priv = dev_get_priv(clk->dev);
487
488 div = meson_clk_get_info(clk, id, MESON_CLK_DIV);
489 if (IS_ERR(div))
490 return PTR_ERR(div);
491
492 mux = meson_clk_get_info(clk, div->parents[0], MESON_CLK_MUX);
493 if (IS_ERR(mux))
494 return PTR_ERR(mux);
495
496 best_parent = -EINVAL;
497 best_delta = ULONG_MAX;
498 for (i = 0; i < (1 << mux->parm->width); i++) {
499 unsigned long parent_rate, delta;
500 unsigned int div_val;
501
502 parent_rate = meson_clk_get_rate_by_id(clk, mux->parents[i]);
503 if (IS_ERR_VALUE(parent_rate))
504 continue;
505
506 /* If overflow, try to use max divider value */
507 div_val = min(DIV_ROUND_CLOSEST(parent_rate, rate),
508 (1UL << div->parm->width));
509
510 delta = abs(rate - (parent_rate / div_val));
511 if (delta < best_delta) {
512 best_delta = delta;
513 best_div_val = div_val;
514 best_parent = i;
515 }
516 }
517
518 if (IS_ERR_VALUE(best_parent))
519 return best_parent;
520
521 SET_PARM_VALUE(priv, mux->parm, best_parent);
522 /* Divider is set to (field value + 1), hence the decrement */
523 SET_PARM_VALUE(priv, div->parm, best_div_val - 1);
524
525 return 0;
526}
527
528static ulong meson_clk_set_rate_by_id(struct clk *clk, unsigned int id, ulong rate);
529
530static ulong meson_mux_set_rate(struct clk *clk, unsigned long id, ulong rate)
531{
532 int i;
533 ulong ret = -EINVAL;
534 struct meson_clk *priv = dev_get_priv(clk->dev);
535 const struct meson_clk_info *info;
536
537 info = meson_clk_get_info(clk, id, MESON_CLK_MUX);
538 if (IS_ERR(info))
539 return PTR_ERR(info);
540
541 for (i = 0; i < (1 << info->parm->width); i++) {
542 ret = meson_clk_set_rate_by_id(clk, info->parents[i], rate);
543 if (!ret) {
544 SET_PARM_VALUE(priv, info->parm, i);
545 break;
546 }
547 }
548
549 return ret;
550}
551
552/* Rate propagation is implemented for a subcection of a clock tree, that is
553 * required at boot stage.
554 */
555static ulong meson_clk_set_rate_by_id(struct clk *clk, unsigned int id, ulong rate)
556{
557 switch (id) {
558 case CLKID_SPIFC_DIV:
559 case CLKID_USB_BUS_DIV:
560 return meson_composite_set_rate(clk, id, rate);
561 case CLKID_SPIFC:
562 case CLKID_USB_BUS: {
563 unsigned long parent = meson_clk_get_parent(clk, id);
564
565 return meson_clk_set_rate_by_id(clk, parent, rate);
566 }
567 case CLKID_SPIFC_SEL2:
568 return meson_mux_set_rate(clk, id, rate);
569 }
570
571 return -EINVAL;
572}
573
574static ulong meson_clk_set_rate(struct clk *clk, ulong rate)
575{
576 return meson_clk_set_rate_by_id(clk, clk->id, rate);
577}
578
579static int meson_mux_set_parent_by_id(struct clk *clk, unsigned int parent_id)
580{
581 unsigned int i, parent_index;
582 struct meson_clk *priv = dev_get_priv(clk->dev);
583 const struct meson_clk_info *info;
584
585 info = meson_clk_get_info(clk, clk->id, MESON_CLK_MUX);
586 if (IS_ERR(info))
587 return PTR_ERR(info);
588
589 parent_index = -EINVAL;
590 for (i = 0; i < (1 << info->parm->width); i++) {
591 if (parent_id == info->parents[i]) {
592 parent_index = i;
593 break;
594 }
595 }
596
597 if (IS_ERR_VALUE(parent_index))
598 return parent_index;
599
600 SET_PARM_VALUE(priv, info->parm, parent_index);
601
602 return 0;
603}
604
605static int meson_clk_set_parent(struct clk *clk, struct clk *parent_clk)
606{
607 return meson_mux_set_parent_by_id(clk, parent_clk->id);
608}
609
610static struct clk_ops meson_clk_ops = {
611 .disable = meson_clk_disable,
612 .enable = meson_clk_enable,
613 .get_rate = meson_clk_get_rate,
614 .set_rate = meson_clk_set_rate,
615 .set_parent = meson_clk_set_parent,
616};
617
618static int meson_clk_probe(struct udevice *dev)
619{
620 struct meson_clk *priv = dev_get_priv(dev);
621
622 return regmap_init_mem(dev_ofnode(dev), &priv->map);
623}
624
625struct meson_clk_data meson_a1_peripherals_info = {
626 .clocks = meson_clocks,
627 .num_clocks = ARRAY_SIZE(meson_clocks),
628};
629
630struct meson_clk_data meson_a1_pll_info = {
631 .clocks = meson_pll_clocks,
632 .num_clocks = ARRAY_SIZE(meson_pll_clocks),
633};
634
635static const struct udevice_id meson_clk_ids[] = {
636 {
637 .compatible = "amlogic,a1-peripherals-clkc",
638 .data = (ulong)&meson_a1_peripherals_info,
639 },
640 {
641 .compatible = "amlogic,a1-pll-clkc",
642 .data = (ulong)&meson_a1_pll_info,
643 },
644 { }
645};
646
647U_BOOT_DRIVER(meson_clk) = {
648 .name = "meson-clk-a1",
649 .id = UCLASS_CLK,
650 .of_match = meson_clk_ids,
651 .priv_auto = sizeof(struct meson_clk),
652 .ops = &meson_clk_ops,
653 .probe = meson_clk_probe,
654};
655
656static const char *meson_clk_get_name(struct clk *clk, int id)
657{
658 const struct meson_clk_info *info;
659
660 info = meson_clk_get_info(clk, id, MESON_CLK_ANY);
661
662 return IS_ERR(info) ? "unknown" : info->name;
663}
664
665static int meson_clk_dump(struct clk *clk)
666{
667 const struct meson_clk_info *info;
668 struct meson_clk *priv;
669 unsigned long rate;
670 char *state, frequency[80];
671 int parent;
672
673 priv = dev_get_priv(clk->dev);
674
675 info = meson_clk_get_info(clk, clk->id, MESON_CLK_ANY);
676 if (IS_ERR(info) || !info->name)
677 return -EINVAL;
678
679 rate = clk_get_rate(clk);
680 if (IS_ERR_VALUE(rate))
681 sprintf(frequency, "unknown");
682 else
683 sprintf(frequency, "%lu", rate);
684
685 if (info->type == MESON_CLK_GATE)
686 state = GET_PARM_VALUE(priv, info->parm) ? "enabled" : "disabled";
687 else
688 state = "N/A";
689
690 parent = meson_clk_get_parent(clk, clk->id);
691 printf("%15s%20s%20s%15s\n",
692 info->name,
693 frequency,
694 meson_clk_get_name(clk, parent),
695 state);
696
697 return 0;
698}
699
700static int meson_clk_dump_dev(struct udevice *dev)
701{
702 int i;
703 struct meson_clk_data *data;
704 const char *sep = "--------------------";
705
706 printf("%s:\n", dev->name);
707 printf("%.15s%s%s%.15s\n", sep, sep, sep, sep);
708 printf("%15s%20s%20s%15s\n", "clk", "frequency", "parent", "state");
709 printf("%.15s%s%s%.15s\n", sep, sep, sep, sep);
710
711 data = (struct meson_clk_data *)dev_get_driver_data(dev);
712 for (i = 0; i < data->num_clocks; i++) {
713 meson_clk_dump(&(struct clk){
714 .dev = dev,
715 .id = i
716 });
717 }
718
719 return 0;
720}
721
722int soc_clk_dump(void)
723{
724 struct udevice *dev;
725 int i = 0;
726
727 while (!uclass_get_device(UCLASS_CLK, i++, &dev)) {
728 if (dev->driver == DM_DRIVER_GET(meson_clk)) {
729 meson_clk_dump_dev(dev);
730 printf("\n");
731 }
732 }
733
734 return 0;
735}