2 * Copyright (c) 2014 Marvell Technology Group Ltd.
4 * Sebastian Hesselbarth <sebastian.hesselbarth@gmail.com>
5 * Alexandre Belloni <alexandre.belloni@free-electrons.com>
7 * This program is free software; you can redistribute it and/or modify it
8 * under the terms and conditions of the GNU General Public License,
9 * version 2, as published by the Free Software Foundation.
11 * This program is distributed in the hope it will be useful, but WITHOUT
12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
16 * You should have received a copy of the GNU General Public License along with
17 * this program. If not, see <http://www.gnu.org/licenses/>.
20 #include <linux/clk.h>
21 #include <linux/clk-provider.h>
22 #include <linux/kernel.h>
24 #include <linux/of_address.h>
25 #include <linux/slab.h>
27 #include <dt-bindings/clock/berlin2.h>
29 #include "berlin2-avpll.h"
30 #include "berlin2-div.h"
31 #include "berlin2-pll.h"
34 #define REG_PINMUX0 0x0000
35 #define REG_PINMUX1 0x0004
36 #define REG_SYSPLLCTL0 0x0014
37 #define REG_SYSPLLCTL4 0x0024
38 #define REG_MEMPLLCTL0 0x0028
39 #define REG_MEMPLLCTL4 0x0038
40 #define REG_CPUPLLCTL0 0x003c
41 #define REG_CPUPLLCTL4 0x004c
42 #define REG_AVPLLCTL0 0x0050
43 #define REG_AVPLLCTL31 0x00cc
44 #define REG_AVPLLCTL62 0x0148
45 #define REG_PLLSTATUS 0x014c
46 #define REG_CLKENABLE 0x0150
47 #define REG_CLKSELECT0 0x0154
48 #define REG_CLKSELECT1 0x0158
49 #define REG_CLKSELECT2 0x015c
50 #define REG_CLKSELECT3 0x0160
51 #define REG_CLKSWITCH0 0x0164
52 #define REG_CLKSWITCH1 0x0168
53 #define REG_RESET_TRIGGER 0x0178
54 #define REG_RESET_STATUS0 0x017c
55 #define REG_RESET_STATUS1 0x0180
56 #define REG_SW_GENERIC0 0x0184
57 #define REG_SW_GENERIC3 0x0190
58 #define REG_PRODUCTID 0x01cc
59 #define REG_PRODUCTID_EXT 0x01d0
60 #define REG_GFX3DCORE_CLKCTL 0x022c
61 #define REG_GFX3DSYS_CLKCTL 0x0230
62 #define REG_ARC_CLKCTL 0x0234
63 #define REG_VIP_CLKCTL 0x0238
64 #define REG_SDIO0XIN_CLKCTL 0x023c
65 #define REG_SDIO1XIN_CLKCTL 0x0240
66 #define REG_GFX3DEXTRA_CLKCTL 0x0244
67 #define REG_GFX3D_RESET 0x0248
68 #define REG_GC360_CLKCTL 0x024c
69 #define REG_SDIO_DLLMST_CLKCTL 0x0250
72 * BG2/BG2CD SoCs have the following audio/video I/O units:
74 * audiohd: HDMI TX audio
80 * video1: Secondary video
81 * video2: SD auxiliary video
83 * There are no external audio clocks (ACLKI0, ACLKI1) and
84 * only one external video clock (VCLKI0).
86 * Currently missing bits and pieces:
87 * - audio_fast_pll is unknown
88 * - audiohd_pll is unknown
89 * - video0_pll is unknown
90 * - audio[023], audiohd parent pll is assumed to be audio_fast_pll
95 static struct clk
*clks
[MAX_CLKS
];
96 static struct clk_onecell_data clk_data
;
97 static DEFINE_SPINLOCK(lock
);
98 static void __iomem
*gbase
;
102 SYSPLL
, MEMPLL
, CPUPLL
,
103 AVPLL_A1
, AVPLL_A2
, AVPLL_A3
, AVPLL_A4
,
104 AVPLL_A5
, AVPLL_A6
, AVPLL_A7
, AVPLL_A8
,
105 AVPLL_B1
, AVPLL_B2
, AVPLL_B3
, AVPLL_B4
,
106 AVPLL_B5
, AVPLL_B6
, AVPLL_B7
, AVPLL_B8
,
107 AUDIO1_PLL
, AUDIO_FAST_PLL
,
108 VIDEO0_PLL
, VIDEO0_IN
,
109 VIDEO1_PLL
, VIDEO1_IN
,
110 VIDEO2_PLL
, VIDEO2_IN
,
113 static const char *clk_names
[] = {
115 [VIDEO_EXT0
] = "video_ext0",
119 [AVPLL_A1
] = "avpll_a1",
120 [AVPLL_A2
] = "avpll_a2",
121 [AVPLL_A3
] = "avpll_a3",
122 [AVPLL_A4
] = "avpll_a4",
123 [AVPLL_A5
] = "avpll_a5",
124 [AVPLL_A6
] = "avpll_a6",
125 [AVPLL_A7
] = "avpll_a7",
126 [AVPLL_A8
] = "avpll_a8",
127 [AVPLL_B1
] = "avpll_b1",
128 [AVPLL_B2
] = "avpll_b2",
129 [AVPLL_B3
] = "avpll_b3",
130 [AVPLL_B4
] = "avpll_b4",
131 [AVPLL_B5
] = "avpll_b5",
132 [AVPLL_B6
] = "avpll_b6",
133 [AVPLL_B7
] = "avpll_b7",
134 [AVPLL_B8
] = "avpll_b8",
135 [AUDIO1_PLL
] = "audio1_pll",
136 [AUDIO_FAST_PLL
] = "audio_fast_pll",
137 [VIDEO0_PLL
] = "video0_pll",
138 [VIDEO0_IN
] = "video0_in",
139 [VIDEO1_PLL
] = "video1_pll",
140 [VIDEO1_IN
] = "video1_in",
141 [VIDEO2_PLL
] = "video2_pll",
142 [VIDEO2_IN
] = "video2_in",
145 static const struct berlin2_pll_map bg2_pll_map __initconst
= {
146 .vcodiv
= {10, 15, 20, 25, 30, 40, 50, 60, 80},
153 static const u8 default_parent_ids
[] = {
154 SYSPLL
, AVPLL_B4
, AVPLL_A5
, AVPLL_B6
, AVPLL_B7
, SYSPLL
157 static const struct berlin2_div_data bg2_divs
[] __initconst
= {
160 .parent_ids
= (const u8
[]){
161 SYSPLL
, AVPLL_B4
, AVPLL_B5
, AVPLL_B6
, AVPLL_B7
, SYSPLL
165 BERLIN2_DIV_GATE(REG_CLKENABLE
, 0),
166 BERLIN2_PLL_SELECT(REG_CLKSELECT0
, 0),
167 BERLIN2_DIV_SELECT(REG_CLKSELECT0
, 3),
168 BERLIN2_PLL_SWITCH(REG_CLKSWITCH0
, 3),
169 BERLIN2_DIV_SWITCH(REG_CLKSWITCH0
, 4),
170 BERLIN2_DIV_D3SWITCH(REG_CLKSWITCH0
, 5),
172 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
173 .flags
= CLK_IGNORE_UNUSED
,
177 .parent_ids
= (const u8
[]){
178 CPUPLL
, MEMPLL
, MEMPLL
, MEMPLL
, MEMPLL
182 BERLIN2_PLL_SELECT(REG_CLKSELECT0
, 6),
183 BERLIN2_DIV_SELECT(REG_CLKSELECT0
, 9),
184 BERLIN2_PLL_SWITCH(REG_CLKSWITCH0
, 6),
185 BERLIN2_DIV_SWITCH(REG_CLKSWITCH0
, 7),
186 BERLIN2_DIV_D3SWITCH(REG_CLKSWITCH0
, 8),
188 .div_flags
= BERLIN2_DIV_HAS_MUX
,
193 .parent_ids
= default_parent_ids
,
194 .num_parents
= ARRAY_SIZE(default_parent_ids
),
196 BERLIN2_DIV_GATE(REG_CLKENABLE
, 16),
197 BERLIN2_PLL_SELECT(REG_CLKSELECT0
, 17),
198 BERLIN2_DIV_SELECT(REG_CLKSELECT0
, 20),
199 BERLIN2_PLL_SWITCH(REG_CLKSWITCH0
, 12),
200 BERLIN2_DIV_SWITCH(REG_CLKSWITCH0
, 13),
201 BERLIN2_DIV_D3SWITCH(REG_CLKSWITCH0
, 14),
203 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
208 .parent_ids
= default_parent_ids
,
209 .num_parents
= ARRAY_SIZE(default_parent_ids
),
211 BERLIN2_DIV_GATE(REG_CLKENABLE
, 1),
212 BERLIN2_PLL_SELECT(REG_CLKSELECT0
, 23),
213 BERLIN2_DIV_SELECT(REG_CLKSELECT0
, 26),
214 BERLIN2_PLL_SWITCH(REG_CLKSWITCH0
, 15),
215 BERLIN2_DIV_SWITCH(REG_CLKSWITCH0
, 16),
216 BERLIN2_DIV_D3SWITCH(REG_CLKSWITCH0
, 17),
218 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
223 .parent_ids
= default_parent_ids
,
224 .num_parents
= ARRAY_SIZE(default_parent_ids
),
226 BERLIN2_DIV_GATE(REG_CLKENABLE
, 4),
227 BERLIN2_PLL_SELECT(REG_CLKSELECT0
, 29),
228 BERLIN2_DIV_SELECT(REG_CLKSELECT1
, 0),
229 BERLIN2_PLL_SWITCH(REG_CLKSWITCH0
, 18),
230 BERLIN2_DIV_SWITCH(REG_CLKSWITCH0
, 19),
231 BERLIN2_DIV_D3SWITCH(REG_CLKSWITCH0
, 20),
233 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
238 .parent_ids
= default_parent_ids
,
239 .num_parents
= ARRAY_SIZE(default_parent_ids
),
241 BERLIN2_DIV_GATE(REG_CLKENABLE
, 5),
242 BERLIN2_PLL_SELECT(REG_CLKSELECT1
, 3),
243 BERLIN2_DIV_SELECT(REG_CLKSELECT1
, 6),
244 BERLIN2_PLL_SWITCH(REG_CLKSWITCH0
, 21),
245 BERLIN2_DIV_SWITCH(REG_CLKSWITCH0
, 22),
246 BERLIN2_DIV_D3SWITCH(REG_CLKSWITCH0
, 23),
248 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
253 .parent_ids
= default_parent_ids
,
254 .num_parents
= ARRAY_SIZE(default_parent_ids
),
256 BERLIN2_DIV_GATE(REG_CLKENABLE
, 6),
257 BERLIN2_PLL_SELECT(REG_CLKSELECT1
, 9),
258 BERLIN2_DIV_SELECT(REG_CLKSELECT1
, 12),
259 BERLIN2_PLL_SWITCH(REG_CLKSWITCH0
, 24),
260 BERLIN2_DIV_SWITCH(REG_CLKSWITCH0
, 25),
261 BERLIN2_DIV_D3SWITCH(REG_CLKSWITCH0
, 26),
263 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
264 .flags
= CLK_IGNORE_UNUSED
,
268 .parent_ids
= default_parent_ids
,
269 .num_parents
= ARRAY_SIZE(default_parent_ids
),
271 BERLIN2_DIV_GATE(REG_CLKENABLE
, 2),
272 BERLIN2_PLL_SELECT(REG_CLKSELECT1
, 15),
273 BERLIN2_DIV_SELECT(REG_CLKSELECT1
, 18),
274 BERLIN2_PLL_SWITCH(REG_CLKSWITCH0
, 27),
275 BERLIN2_DIV_SWITCH(REG_CLKSWITCH0
, 28),
276 BERLIN2_DIV_D3SWITCH(REG_CLKSWITCH0
, 29),
278 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
283 .parent_ids
= default_parent_ids
,
284 .num_parents
= ARRAY_SIZE(default_parent_ids
),
286 BERLIN2_DIV_GATE(REG_CLKENABLE
, 3),
287 BERLIN2_PLL_SELECT(REG_CLKSELECT1
, 21),
288 BERLIN2_DIV_SELECT(REG_CLKSELECT1
, 24),
289 BERLIN2_PLL_SWITCH(REG_CLKSWITCH0
, 30),
290 BERLIN2_DIV_SWITCH(REG_CLKSWITCH0
, 31),
291 BERLIN2_DIV_D3SWITCH(REG_CLKSWITCH1
, 0),
293 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
298 .parent_ids
= default_parent_ids
,
299 .num_parents
= ARRAY_SIZE(default_parent_ids
),
301 BERLIN2_DIV_GATE(REG_CLKENABLE
, 18),
302 BERLIN2_PLL_SELECT(REG_CLKSELECT1
, 27),
303 BERLIN2_DIV_SELECT(REG_CLKSELECT2
, 0),
304 BERLIN2_PLL_SWITCH(REG_CLKSWITCH1
, 1),
305 BERLIN2_DIV_SWITCH(REG_CLKSWITCH1
, 2),
306 BERLIN2_DIV_D3SWITCH(REG_CLKSWITCH1
, 3),
308 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
313 .parent_ids
= default_parent_ids
,
314 .num_parents
= ARRAY_SIZE(default_parent_ids
),
316 BERLIN2_DIV_GATE(REG_CLKENABLE
, 21),
317 BERLIN2_PLL_SELECT(REG_CLKSELECT2
, 3),
318 BERLIN2_DIV_SELECT(REG_CLKSELECT2
, 6),
319 BERLIN2_PLL_SWITCH(REG_CLKSWITCH1
, 4),
320 BERLIN2_DIV_SWITCH(REG_CLKSWITCH1
, 5),
321 BERLIN2_DIV_D3SWITCH(REG_CLKSWITCH1
, 6),
323 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
328 .parent_ids
= default_parent_ids
,
329 .num_parents
= ARRAY_SIZE(default_parent_ids
),
331 BERLIN2_DIV_GATE(REG_CLKENABLE
, 20),
332 BERLIN2_PLL_SELECT(REG_CLKSELECT2
, 9),
333 BERLIN2_DIV_SELECT(REG_CLKSELECT2
, 12),
334 BERLIN2_PLL_SWITCH(REG_CLKSWITCH1
, 7),
335 BERLIN2_DIV_SWITCH(REG_CLKSWITCH1
, 8),
336 BERLIN2_DIV_D3SWITCH(REG_CLKSWITCH1
, 9),
338 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
343 .parent_ids
= (const u8
[]){ AUDIO_FAST_PLL
},
346 BERLIN2_DIV_GATE(REG_CLKENABLE
, 22),
347 BERLIN2_DIV_SELECT(REG_CLKSELECT2
, 17),
348 BERLIN2_DIV_SWITCH(REG_CLKSWITCH1
, 10),
349 BERLIN2_DIV_D3SWITCH(REG_CLKSWITCH1
, 11),
351 .div_flags
= BERLIN2_DIV_HAS_GATE
,
356 .parent_ids
= (const u8
[]){ AUDIO_FAST_PLL
},
359 BERLIN2_DIV_GATE(REG_CLKENABLE
, 24),
360 BERLIN2_DIV_SELECT(REG_CLKSELECT2
, 20),
361 BERLIN2_DIV_SWITCH(REG_CLKSWITCH1
, 14),
362 BERLIN2_DIV_D3SWITCH(REG_CLKSWITCH1
, 15),
364 .div_flags
= BERLIN2_DIV_HAS_GATE
,
369 .parent_ids
= (const u8
[]){ AUDIO_FAST_PLL
},
372 BERLIN2_DIV_GATE(REG_CLKENABLE
, 25),
373 BERLIN2_DIV_SELECT(REG_CLKSELECT2
, 23),
374 BERLIN2_DIV_SWITCH(REG_CLKSWITCH1
, 16),
375 BERLIN2_DIV_D3SWITCH(REG_CLKSWITCH1
, 17),
377 .div_flags
= BERLIN2_DIV_HAS_GATE
,
382 .parent_ids
= (const u8
[]){ AUDIO1_PLL
},
385 BERLIN2_DIV_GATE(REG_CLKENABLE
, 23),
386 BERLIN2_DIV_SELECT(REG_CLKSELECT3
, 0),
387 BERLIN2_DIV_SWITCH(REG_CLKSWITCH1
, 12),
388 BERLIN2_DIV_D3SWITCH(REG_CLKSWITCH1
, 13),
390 .div_flags
= BERLIN2_DIV_HAS_GATE
,
394 .name
= "gfx3d_core",
395 .parent_ids
= default_parent_ids
,
396 .num_parents
= ARRAY_SIZE(default_parent_ids
),
398 BERLIN2_SINGLE_DIV(REG_GFX3DCORE_CLKCTL
),
400 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
405 .parent_ids
= default_parent_ids
,
406 .num_parents
= ARRAY_SIZE(default_parent_ids
),
408 BERLIN2_SINGLE_DIV(REG_GFX3DSYS_CLKCTL
),
410 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
415 .parent_ids
= default_parent_ids
,
416 .num_parents
= ARRAY_SIZE(default_parent_ids
),
418 BERLIN2_SINGLE_DIV(REG_ARC_CLKCTL
),
420 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
425 .parent_ids
= default_parent_ids
,
426 .num_parents
= ARRAY_SIZE(default_parent_ids
),
428 BERLIN2_SINGLE_DIV(REG_VIP_CLKCTL
),
430 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
435 .parent_ids
= default_parent_ids
,
436 .num_parents
= ARRAY_SIZE(default_parent_ids
),
438 BERLIN2_SINGLE_DIV(REG_SDIO0XIN_CLKCTL
),
440 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
445 .parent_ids
= default_parent_ids
,
446 .num_parents
= ARRAY_SIZE(default_parent_ids
),
448 BERLIN2_SINGLE_DIV(REG_SDIO1XIN_CLKCTL
),
450 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
454 .name
= "gfx3d_extra",
455 .parent_ids
= default_parent_ids
,
456 .num_parents
= ARRAY_SIZE(default_parent_ids
),
458 BERLIN2_SINGLE_DIV(REG_GFX3DEXTRA_CLKCTL
),
460 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
465 .parent_ids
= default_parent_ids
,
466 .num_parents
= ARRAY_SIZE(default_parent_ids
),
468 BERLIN2_SINGLE_DIV(REG_GC360_CLKCTL
),
470 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
474 .name
= "sdio_dllmst",
475 .parent_ids
= default_parent_ids
,
476 .num_parents
= ARRAY_SIZE(default_parent_ids
),
478 BERLIN2_SINGLE_DIV(REG_SDIO_DLLMST_CLKCTL
),
480 .div_flags
= BERLIN2_DIV_HAS_GATE
| BERLIN2_DIV_HAS_MUX
,
485 static const struct berlin2_gate_data bg2_gates
[] __initconst
= {
486 { "geth0", "perif", 7 },
487 { "geth1", "perif", 8 },
488 { "sata", "perif", 9 },
489 { "ahbapb", "perif", 10, CLK_IGNORE_UNUSED
},
490 { "usb0", "perif", 11 },
491 { "usb1", "perif", 12 },
492 { "pbridge", "perif", 13, CLK_IGNORE_UNUSED
},
493 { "sdio0", "perif", 14, CLK_IGNORE_UNUSED
},
494 { "sdio1", "perif", 15, CLK_IGNORE_UNUSED
},
495 { "nfc", "perif", 17 },
496 { "smemc", "perif", 19 },
497 { "audiohd", "audiohd_pll", 26 },
498 { "video0", "video0_in", 27 },
499 { "video1", "video1_in", 28 },
500 { "video2", "video2_in", 29 },
503 static void __init
berlin2_clock_setup(struct device_node
*np
)
505 struct device_node
*parent_np
= of_get_parent(np
);
506 const char *parent_names
[9];
511 gbase
= of_iomap(parent_np
, 0);
515 /* overwrite default clock names with DT provided ones */
516 clk
= of_clk_get_by_name(np
, clk_names
[REFCLK
]);
518 clk_names
[REFCLK
] = __clk_get_name(clk
);
522 clk
= of_clk_get_by_name(np
, clk_names
[VIDEO_EXT0
]);
524 clk_names
[VIDEO_EXT0
] = __clk_get_name(clk
);
528 /* simple register PLLs */
529 clk
= berlin2_pll_register(&bg2_pll_map
, gbase
+ REG_SYSPLLCTL0
,
530 clk_names
[SYSPLL
], clk_names
[REFCLK
], 0);
534 clk
= berlin2_pll_register(&bg2_pll_map
, gbase
+ REG_MEMPLLCTL0
,
535 clk_names
[MEMPLL
], clk_names
[REFCLK
], 0);
539 clk
= berlin2_pll_register(&bg2_pll_map
, gbase
+ REG_CPUPLLCTL0
,
540 clk_names
[CPUPLL
], clk_names
[REFCLK
], 0);
544 if (of_device_is_compatible(np
, "marvell,berlin2-global-register"))
545 avpll_flags
|= BERLIN2_AVPLL_SCRAMBLE_QUIRK
;
547 /* audio/video VCOs */
548 clk
= berlin2_avpll_vco_register(gbase
+ REG_AVPLLCTL0
, "avpll_vcoA",
549 clk_names
[REFCLK
], avpll_flags
, 0);
553 for (n
= 0; n
< 8; n
++) {
554 clk
= berlin2_avpll_channel_register(gbase
+ REG_AVPLLCTL0
,
555 clk_names
[AVPLL_A1
+ n
], n
, "avpll_vcoA",
561 clk
= berlin2_avpll_vco_register(gbase
+ REG_AVPLLCTL31
, "avpll_vcoB",
562 clk_names
[REFCLK
], BERLIN2_AVPLL_BIT_QUIRK
|
567 for (n
= 0; n
< 8; n
++) {
568 clk
= berlin2_avpll_channel_register(gbase
+ REG_AVPLLCTL31
,
569 clk_names
[AVPLL_B1
+ n
], n
, "avpll_vcoB",
570 BERLIN2_AVPLL_BIT_QUIRK
| avpll_flags
, 0);
575 /* reference clock bypass switches */
576 parent_names
[0] = clk_names
[SYSPLL
];
577 parent_names
[1] = clk_names
[REFCLK
];
578 clk
= clk_register_mux(NULL
, "syspll_byp", parent_names
, 2,
579 0, gbase
+ REG_CLKSWITCH0
, 0, 1, 0, &lock
);
582 clk_names
[SYSPLL
] = __clk_get_name(clk
);
584 parent_names
[0] = clk_names
[MEMPLL
];
585 parent_names
[1] = clk_names
[REFCLK
];
586 clk
= clk_register_mux(NULL
, "mempll_byp", parent_names
, 2,
587 0, gbase
+ REG_CLKSWITCH0
, 1, 1, 0, &lock
);
590 clk_names
[MEMPLL
] = __clk_get_name(clk
);
592 parent_names
[0] = clk_names
[CPUPLL
];
593 parent_names
[1] = clk_names
[REFCLK
];
594 clk
= clk_register_mux(NULL
, "cpupll_byp", parent_names
, 2,
595 0, gbase
+ REG_CLKSWITCH0
, 2, 1, 0, &lock
);
598 clk_names
[CPUPLL
] = __clk_get_name(clk
);
601 parent_names
[0] = clk_names
[AVPLL_B3
];
602 parent_names
[1] = clk_names
[AVPLL_A3
];
603 clk
= clk_register_mux(NULL
, clk_names
[AUDIO1_PLL
], parent_names
, 2,
604 0, gbase
+ REG_CLKSELECT2
, 29, 1, 0, &lock
);
608 parent_names
[0] = clk_names
[VIDEO0_PLL
];
609 parent_names
[1] = clk_names
[VIDEO_EXT0
];
610 clk
= clk_register_mux(NULL
, clk_names
[VIDEO0_IN
], parent_names
, 2,
611 0, gbase
+ REG_CLKSELECT3
, 4, 1, 0, &lock
);
615 parent_names
[0] = clk_names
[VIDEO1_PLL
];
616 parent_names
[1] = clk_names
[VIDEO_EXT0
];
617 clk
= clk_register_mux(NULL
, clk_names
[VIDEO1_IN
], parent_names
, 2,
618 0, gbase
+ REG_CLKSELECT3
, 6, 1, 0, &lock
);
622 parent_names
[0] = clk_names
[AVPLL_A2
];
623 parent_names
[1] = clk_names
[AVPLL_B2
];
624 clk
= clk_register_mux(NULL
, clk_names
[VIDEO1_PLL
], parent_names
, 2,
625 0, gbase
+ REG_CLKSELECT3
, 7, 1, 0, &lock
);
629 parent_names
[0] = clk_names
[VIDEO2_PLL
];
630 parent_names
[1] = clk_names
[VIDEO_EXT0
];
631 clk
= clk_register_mux(NULL
, clk_names
[VIDEO2_IN
], parent_names
, 2,
632 0, gbase
+ REG_CLKSELECT3
, 9, 1, 0, &lock
);
636 parent_names
[0] = clk_names
[AVPLL_B1
];
637 parent_names
[1] = clk_names
[AVPLL_A5
];
638 clk
= clk_register_mux(NULL
, clk_names
[VIDEO2_PLL
], parent_names
, 2,
639 0, gbase
+ REG_CLKSELECT3
, 10, 1, 0, &lock
);
643 /* clock divider cells */
644 for (n
= 0; n
< ARRAY_SIZE(bg2_divs
); n
++) {
645 const struct berlin2_div_data
*dd
= &bg2_divs
[n
];
648 for (k
= 0; k
< dd
->num_parents
; k
++)
649 parent_names
[k
] = clk_names
[dd
->parent_ids
[k
]];
651 clks
[CLKID_SYS
+ n
] = berlin2_div_register(&dd
->map
, gbase
,
652 dd
->name
, dd
->div_flags
, parent_names
,
653 dd
->num_parents
, dd
->flags
, &lock
);
656 /* clock gate cells */
657 for (n
= 0; n
< ARRAY_SIZE(bg2_gates
); n
++) {
658 const struct berlin2_gate_data
*gd
= &bg2_gates
[n
];
660 clks
[CLKID_GETH0
+ n
] = clk_register_gate(NULL
, gd
->name
,
661 gd
->parent_name
, gd
->flags
, gbase
+ REG_CLKENABLE
,
662 gd
->bit_idx
, 0, &lock
);
665 /* twdclk is derived from cpu/3 */
667 clk_register_fixed_factor(NULL
, "twd", "cpu", 0, 1, 3);
669 /* check for errors on leaf clocks */
670 for (n
= 0; n
< MAX_CLKS
; n
++) {
671 if (!IS_ERR(clks
[n
]))
674 pr_err("%s: Unable to register leaf clock %d\n",
679 /* register clk-provider */
680 clk_data
.clks
= clks
;
681 clk_data
.clk_num
= MAX_CLKS
;
682 of_clk_add_provider(np
, of_clk_src_onecell_get
, &clk_data
);
689 CLK_OF_DECLARE(berlin2_clk
, "marvell,berlin2-clk",
690 berlin2_clock_setup
);