1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3 * Copyright (c) 2013 Samsung Electronics Co., Ltd.
4 * Copyright (c) 2013 Linaro Ltd.
5 * Author: Thomas Abraham <thomas.ab@samsung.com>
6 *
7 * This file includes utility functions to register clocks to common
8 * clock framework for Samsung platforms.
9 */
10
11 #include <linux/slab.h>
12 #include <linux/clkdev.h>
13 #include <linux/clk-provider.h>
14 #include <linux/io.h>
15 #include <linux/mfd/syscon.h>
16 #include <linux/mod_devicetable.h>
17 #include <linux/of_address.h>
18 #include <linux/regmap.h>
19 #include <linux/syscore_ops.h>
20
21 #include "clk.h"
22
23 static LIST_HEAD(clock_reg_cache_list);
24
samsung_clk_save(void __iomem * base,struct regmap * regmap,struct samsung_clk_reg_dump * rd,unsigned int num_regs)25 void samsung_clk_save(void __iomem *base,
26 struct regmap *regmap,
27 struct samsung_clk_reg_dump *rd,
28 unsigned int num_regs)
29 {
30 for (; num_regs > 0; --num_regs, ++rd) {
31 if (base)
32 rd->value = readl(base + rd->offset);
33 else if (regmap)
34 regmap_read(regmap, rd->offset, &rd->value);
35 }
36 }
37
samsung_clk_restore(void __iomem * base,struct regmap * regmap,const struct samsung_clk_reg_dump * rd,unsigned int num_regs)38 void samsung_clk_restore(void __iomem *base,
39 struct regmap *regmap,
40 const struct samsung_clk_reg_dump *rd,
41 unsigned int num_regs)
42 {
43 for (; num_regs > 0; --num_regs, ++rd) {
44 if (base)
45 writel(rd->value, base + rd->offset);
46 else if (regmap)
47 regmap_write(regmap, rd->offset, rd->value);
48 }
49 }
50
samsung_clk_alloc_reg_dump(const unsigned long * rdump,unsigned long nr_rdump)51 struct samsung_clk_reg_dump *samsung_clk_alloc_reg_dump(
52 const unsigned long *rdump,
53 unsigned long nr_rdump)
54 {
55 struct samsung_clk_reg_dump *rd;
56 unsigned int i;
57
58 rd = kzalloc_objs(*rd, nr_rdump);
59 if (!rd)
60 return NULL;
61
62 for (i = 0; i < nr_rdump; ++i)
63 rd[i].offset = rdump[i];
64
65 return rd;
66 }
67
68 /**
69 * samsung_clk_init() - Create and initialize a clock provider object
70 * @dev: CMU device to enable runtime PM, or NULL if RPM is not needed
71 * @base: Start address (mapped) of CMU registers
72 * @nr_clks: Total clock count to allocate in clock provider object
73 *
74 * Setup the essentials required to support clock lookup using Common Clock
75 * Framework.
76 *
77 * Return: Allocated and initialized clock provider object.
78 */
samsung_clk_init(struct device * dev,void __iomem * base,unsigned long nr_clks)79 struct samsung_clk_provider * __init samsung_clk_init(struct device *dev,
80 void __iomem *base, unsigned long nr_clks)
81 {
82 struct samsung_clk_provider *ctx;
83 int i;
84
85 ctx = kzalloc_flex(*ctx, clk_data.hws, nr_clks);
86 if (!ctx)
87 panic("could not allocate clock provider context.\n");
88
89 ctx->clk_data.num = nr_clks;
90 for (i = 0; i < nr_clks; ++i)
91 ctx->clk_data.hws[i] = ERR_PTR(-ENOENT);
92
93 ctx->dev = dev;
94 ctx->reg_base = base;
95 spin_lock_init(&ctx->lock);
96
97 return ctx;
98 }
99
samsung_clk_of_add_provider(struct device_node * np,struct samsung_clk_provider * ctx)100 void __init samsung_clk_of_add_provider(struct device_node *np,
101 struct samsung_clk_provider *ctx)
102 {
103 if (np) {
104 if (of_clk_add_hw_provider(np, of_clk_hw_onecell_get,
105 &ctx->clk_data))
106 panic("could not register clk provider\n");
107 }
108 }
109
110 /* add a clock instance to the clock lookup table used for dt based lookup */
samsung_clk_add_lookup(struct samsung_clk_provider * ctx,struct clk_hw * clk_hw,unsigned int id)111 void samsung_clk_add_lookup(struct samsung_clk_provider *ctx,
112 struct clk_hw *clk_hw, unsigned int id)
113 {
114 if (id)
115 ctx->clk_data.hws[id] = clk_hw;
116 }
117
118 /* register a list of aliases */
samsung_clk_register_alias(struct samsung_clk_provider * ctx,const struct samsung_clock_alias * list,unsigned int nr_clk)119 void __init samsung_clk_register_alias(struct samsung_clk_provider *ctx,
120 const struct samsung_clock_alias *list,
121 unsigned int nr_clk)
122 {
123 struct clk_hw *clk_hw;
124 unsigned int idx, ret;
125
126 for (idx = 0; idx < nr_clk; idx++, list++) {
127 if (!list->id) {
128 pr_err("%s: clock id missing for index %d\n", __func__,
129 idx);
130 continue;
131 }
132
133 clk_hw = ctx->clk_data.hws[list->id];
134 if (!clk_hw) {
135 pr_err("%s: failed to find clock %d\n", __func__,
136 list->id);
137 continue;
138 }
139
140 ret = clk_hw_register_clkdev(clk_hw, list->alias,
141 list->dev_name);
142 if (ret)
143 pr_err("%s: failed to register lookup %s\n",
144 __func__, list->alias);
145 }
146 }
147
148 /* register a list of fixed clocks */
samsung_clk_register_fixed_rate(struct samsung_clk_provider * ctx,const struct samsung_fixed_rate_clock * list,unsigned int nr_clk)149 void __init samsung_clk_register_fixed_rate(struct samsung_clk_provider *ctx,
150 const struct samsung_fixed_rate_clock *list,
151 unsigned int nr_clk)
152 {
153 struct clk_hw *clk_hw;
154 unsigned int idx;
155
156 for (idx = 0; idx < nr_clk; idx++, list++) {
157 clk_hw = clk_hw_register_fixed_rate(ctx->dev, list->name,
158 list->parent_name, list->flags, list->fixed_rate);
159 if (IS_ERR(clk_hw)) {
160 pr_err("%s: failed to register clock %s\n", __func__,
161 list->name);
162 continue;
163 }
164
165 samsung_clk_add_lookup(ctx, clk_hw, list->id);
166 }
167 }
168
169 /* register a list of fixed factor clocks */
samsung_clk_register_fixed_factor(struct samsung_clk_provider * ctx,const struct samsung_fixed_factor_clock * list,unsigned int nr_clk)170 void __init samsung_clk_register_fixed_factor(struct samsung_clk_provider *ctx,
171 const struct samsung_fixed_factor_clock *list, unsigned int nr_clk)
172 {
173 struct clk_hw *clk_hw;
174 unsigned int idx;
175
176 for (idx = 0; idx < nr_clk; idx++, list++) {
177 clk_hw = clk_hw_register_fixed_factor(ctx->dev, list->name,
178 list->parent_name, list->flags, list->mult, list->div);
179 if (IS_ERR(clk_hw)) {
180 pr_err("%s: failed to register clock %s\n", __func__,
181 list->name);
182 continue;
183 }
184
185 samsung_clk_add_lookup(ctx, clk_hw, list->id);
186 }
187 }
188
189 /* register a list of mux clocks */
samsung_clk_register_mux(struct samsung_clk_provider * ctx,const struct samsung_mux_clock * list,unsigned int nr_clk)190 void __init samsung_clk_register_mux(struct samsung_clk_provider *ctx,
191 const struct samsung_mux_clock *list,
192 unsigned int nr_clk)
193 {
194 struct clk_hw *clk_hw;
195 unsigned int idx;
196
197 for (idx = 0; idx < nr_clk; idx++, list++) {
198 clk_hw = clk_hw_register_mux(ctx->dev, list->name,
199 list->parent_names, list->num_parents, list->flags,
200 ctx->reg_base + list->offset,
201 list->shift, list->width, list->mux_flags, &ctx->lock);
202 if (IS_ERR(clk_hw)) {
203 pr_err("%s: failed to register clock %s\n", __func__,
204 list->name);
205 continue;
206 }
207
208 samsung_clk_add_lookup(ctx, clk_hw, list->id);
209 }
210 }
211
212 /* register a list of div clocks */
samsung_clk_register_div(struct samsung_clk_provider * ctx,const struct samsung_div_clock * list,unsigned int nr_clk)213 void __init samsung_clk_register_div(struct samsung_clk_provider *ctx,
214 const struct samsung_div_clock *list,
215 unsigned int nr_clk)
216 {
217 struct clk_hw *clk_hw;
218 unsigned int idx;
219
220 for (idx = 0; idx < nr_clk; idx++, list++) {
221 if (list->table)
222 clk_hw = clk_hw_register_divider_table(ctx->dev,
223 list->name, list->parent_name, list->flags,
224 ctx->reg_base + list->offset,
225 list->shift, list->width, list->div_flags,
226 list->table, &ctx->lock);
227 else
228 clk_hw = clk_hw_register_divider(ctx->dev, list->name,
229 list->parent_name, list->flags,
230 ctx->reg_base + list->offset, list->shift,
231 list->width, list->div_flags, &ctx->lock);
232 if (IS_ERR(clk_hw)) {
233 pr_err("%s: failed to register clock %s\n", __func__,
234 list->name);
235 continue;
236 }
237
238 samsung_clk_add_lookup(ctx, clk_hw, list->id);
239 }
240 }
241
242 /*
243 * Some older DT's have an incorrect CMU resource size which is incompatible
244 * with the auto clock mode feature. In such cases we switch back to manual
245 * clock gating mode.
246 */
samsung_is_auto_capable(struct device_node * np)247 bool samsung_is_auto_capable(struct device_node *np)
248 {
249 struct resource res;
250 resource_size_t size;
251
252 if (of_address_to_resource(np, 0, &res))
253 return false;
254
255 size = resource_size(&res);
256 if (size != 0x10000) {
257 pr_warn("%pOF: incorrect res size for automatic clocks\n", np);
258 return false;
259 }
260 return true;
261 }
262
263 #define ACG_MSK GENMASK(6, 4)
264 #define CLK_IDLE GENMASK(5, 4)
samsung_auto_clk_gate_is_en(struct clk_hw * hw)265 static int samsung_auto_clk_gate_is_en(struct clk_hw *hw)
266 {
267 u32 reg;
268 struct clk_gate *gate = to_clk_gate(hw);
269
270 reg = readl(gate->reg);
271 return ((reg & ACG_MSK) == CLK_IDLE) ? 0 : 1;
272 }
273
274 /* enable and disable are nops in automatic clock mode */
samsung_auto_clk_gate_en(struct clk_hw * hw)275 static int samsung_auto_clk_gate_en(struct clk_hw *hw)
276 {
277 return 0;
278 }
279
samsung_auto_clk_gate_dis(struct clk_hw * hw)280 static void samsung_auto_clk_gate_dis(struct clk_hw *hw)
281 {
282 }
283
284 static const struct clk_ops samsung_auto_clk_gate_ops = {
285 .enable = samsung_auto_clk_gate_en,
286 .disable = samsung_auto_clk_gate_dis,
287 .is_enabled = samsung_auto_clk_gate_is_en,
288 };
289
samsung_register_auto_gate(struct device * dev,struct device_node * np,const char * name,const char * parent_name,const struct clk_hw * parent_hw,const struct clk_parent_data * parent_data,unsigned long flags,void __iomem * reg,u8 bit_idx,u8 clk_gate_flags,spinlock_t * lock)290 struct clk_hw *samsung_register_auto_gate(struct device *dev,
291 struct device_node *np, const char *name,
292 const char *parent_name, const struct clk_hw *parent_hw,
293 const struct clk_parent_data *parent_data,
294 unsigned long flags,
295 void __iomem *reg, u8 bit_idx,
296 u8 clk_gate_flags, spinlock_t *lock)
297 {
298 struct clk_gate *gate;
299 struct clk_hw *hw;
300 struct clk_init_data init = {};
301 int ret = -EINVAL;
302
303 /* allocate the gate */
304 gate = kzalloc_obj(*gate);
305 if (!gate)
306 return ERR_PTR(-ENOMEM);
307
308 init.name = name;
309 init.ops = &samsung_auto_clk_gate_ops;
310 init.flags = flags;
311 init.parent_names = parent_name ? &parent_name : NULL;
312 init.parent_hws = parent_hw ? &parent_hw : NULL;
313 init.parent_data = parent_data;
314 if (parent_name || parent_hw || parent_data)
315 init.num_parents = 1;
316 else
317 init.num_parents = 0;
318
319 /* struct clk_gate assignments */
320 gate->reg = reg;
321 gate->bit_idx = bit_idx;
322 gate->flags = clk_gate_flags;
323 gate->lock = lock;
324 gate->hw.init = &init;
325
326 hw = &gate->hw;
327 if (dev || !np)
328 ret = clk_hw_register(dev, hw);
329 else if (np)
330 ret = of_clk_hw_register(np, hw);
331 if (ret) {
332 kfree(gate);
333 hw = ERR_PTR(ret);
334 }
335
336 return hw;
337 }
338
339 /* register a list of gate clocks */
samsung_clk_register_gate(struct samsung_clk_provider * ctx,const struct samsung_gate_clock * list,unsigned int nr_clk)340 void __init samsung_clk_register_gate(struct samsung_clk_provider *ctx,
341 const struct samsung_gate_clock *list,
342 unsigned int nr_clk)
343 {
344 struct clk_hw *clk_hw;
345 unsigned int idx;
346 void __iomem *reg_offs;
347
348 for (idx = 0; idx < nr_clk; idx++, list++) {
349 reg_offs = ctx->reg_base + list->offset;
350
351 if (ctx->auto_clock_gate && ctx->gate_dbg_offset)
352 clk_hw = samsung_register_auto_gate(ctx->dev, NULL,
353 list->name, list->parent_name, NULL, NULL,
354 list->flags, reg_offs + ctx->gate_dbg_offset,
355 list->bit_idx, list->gate_flags, &ctx->lock);
356 else
357 clk_hw = clk_hw_register_gate(ctx->dev, list->name,
358 list->parent_name, list->flags,
359 ctx->reg_base + list->offset, list->bit_idx,
360 list->gate_flags, &ctx->lock);
361 if (IS_ERR(clk_hw)) {
362 pr_err("%s: failed to register clock %s: %ld\n", __func__,
363 list->name, PTR_ERR(clk_hw));
364 continue;
365 }
366
367 samsung_clk_add_lookup(ctx, clk_hw, list->id);
368 }
369 }
370
371 /*
372 * obtain the clock speed of all external fixed clock sources from device
373 * tree and register it
374 */
samsung_clk_of_register_fixed_ext(struct samsung_clk_provider * ctx,struct samsung_fixed_rate_clock * fixed_rate_clk,unsigned int nr_fixed_rate_clk,const struct of_device_id * clk_matches)375 void __init samsung_clk_of_register_fixed_ext(struct samsung_clk_provider *ctx,
376 struct samsung_fixed_rate_clock *fixed_rate_clk,
377 unsigned int nr_fixed_rate_clk,
378 const struct of_device_id *clk_matches)
379 {
380 const struct of_device_id *match;
381 struct device_node *clk_np;
382 u32 freq;
383
384 for_each_matching_node_and_match(clk_np, clk_matches, &match) {
385 if (of_property_read_u32(clk_np, "clock-frequency", &freq))
386 continue;
387 fixed_rate_clk[(unsigned long)match->data].fixed_rate = freq;
388 }
389 samsung_clk_register_fixed_rate(ctx, fixed_rate_clk, nr_fixed_rate_clk);
390 }
391
392 #ifdef CONFIG_PM_SLEEP
samsung_clk_suspend(void * data)393 static int samsung_clk_suspend(void *data)
394 {
395 struct samsung_clock_reg_cache *reg_cache;
396
397 list_for_each_entry(reg_cache, &clock_reg_cache_list, node) {
398 samsung_clk_save(reg_cache->reg_base, reg_cache->sysreg,
399 reg_cache->rdump, reg_cache->rd_num);
400 samsung_clk_restore(reg_cache->reg_base, reg_cache->sysreg,
401 reg_cache->rsuspend,
402 reg_cache->rsuspend_num);
403 }
404 return 0;
405 }
406
samsung_clk_resume(void * data)407 static void samsung_clk_resume(void *data)
408 {
409 struct samsung_clock_reg_cache *reg_cache;
410
411 list_for_each_entry(reg_cache, &clock_reg_cache_list, node)
412 samsung_clk_restore(reg_cache->reg_base, reg_cache->sysreg,
413 reg_cache->rdump, reg_cache->rd_num);
414 }
415
416 static const struct syscore_ops samsung_clk_syscore_ops = {
417 .suspend = samsung_clk_suspend,
418 .resume = samsung_clk_resume,
419 };
420
421 static struct syscore samsung_clk_syscore = {
422 .ops = &samsung_clk_syscore_ops,
423 };
424
samsung_clk_extended_sleep_init(void __iomem * reg_base,struct regmap * sysreg,const unsigned long * rdump,unsigned long nr_rdump,const struct samsung_clk_reg_dump * rsuspend,unsigned long nr_rsuspend)425 void samsung_clk_extended_sleep_init(void __iomem *reg_base,
426 struct regmap *sysreg,
427 const unsigned long *rdump,
428 unsigned long nr_rdump,
429 const struct samsung_clk_reg_dump *rsuspend,
430 unsigned long nr_rsuspend)
431 {
432 struct samsung_clock_reg_cache *reg_cache;
433
434 reg_cache = kzalloc_obj(struct samsung_clock_reg_cache);
435 if (!reg_cache)
436 panic("could not allocate register reg_cache.\n");
437 reg_cache->rdump = samsung_clk_alloc_reg_dump(rdump, nr_rdump);
438
439 if (!reg_cache->rdump)
440 panic("could not allocate register dump storage.\n");
441
442 if (list_empty(&clock_reg_cache_list))
443 register_syscore(&samsung_clk_syscore);
444
445 reg_cache->reg_base = reg_base;
446 reg_cache->sysreg = sysreg;
447 reg_cache->rd_num = nr_rdump;
448 reg_cache->rsuspend = rsuspend;
449 reg_cache->rsuspend_num = nr_rsuspend;
450 list_add_tail(®_cache->node, &clock_reg_cache_list);
451 }
452 #endif
453
454 /**
455 * samsung_cmu_register_clocks() - Register all clocks provided in CMU object
456 * @ctx: Clock provider object
457 * @cmu: CMU object with clocks to register
458 * @np: CMU device tree node
459 */
samsung_cmu_register_clocks(struct samsung_clk_provider * ctx,const struct samsung_cmu_info * cmu,struct device_node * np)460 void __init samsung_cmu_register_clocks(struct samsung_clk_provider *ctx,
461 const struct samsung_cmu_info *cmu,
462 struct device_node *np)
463 {
464 if (cmu->auto_clock_gate && samsung_is_auto_capable(np))
465 ctx->auto_clock_gate = cmu->auto_clock_gate;
466
467 ctx->gate_dbg_offset = cmu->gate_dbg_offset;
468 ctx->option_offset = cmu->option_offset;
469 ctx->drcg_offset = cmu->drcg_offset;
470 ctx->memclk_offset = cmu->memclk_offset;
471
472 if (cmu->pll_clks)
473 samsung_clk_register_pll(ctx, cmu->pll_clks, cmu->nr_pll_clks);
474 if (cmu->mux_clks)
475 samsung_clk_register_mux(ctx, cmu->mux_clks, cmu->nr_mux_clks);
476 if (cmu->div_clks)
477 samsung_clk_register_div(ctx, cmu->div_clks, cmu->nr_div_clks);
478 if (cmu->gate_clks)
479 samsung_clk_register_gate(ctx, cmu->gate_clks,
480 cmu->nr_gate_clks);
481 if (cmu->fixed_clks)
482 samsung_clk_register_fixed_rate(ctx, cmu->fixed_clks,
483 cmu->nr_fixed_clks);
484 if (cmu->fixed_factor_clks)
485 samsung_clk_register_fixed_factor(ctx, cmu->fixed_factor_clks,
486 cmu->nr_fixed_factor_clks);
487 if (cmu->cpu_clks)
488 samsung_clk_register_cpu(ctx, cmu->cpu_clks, cmu->nr_cpu_clks);
489 }
490
491 /* Each bit enable/disables DRCG of a bus component */
492 #define DRCG_EN_MSK GENMASK(31, 0)
493 #define MEMCLK_EN BIT(0)
494
495 /* Enable Dynamic Root Clock Gating (DRCG) of bus components */
samsung_en_dyn_root_clk_gating(struct device_node * np,struct samsung_clk_provider * ctx,const struct samsung_cmu_info * cmu,bool cmu_has_pm)496 void samsung_en_dyn_root_clk_gating(struct device_node *np,
497 struct samsung_clk_provider *ctx,
498 const struct samsung_cmu_info *cmu,
499 bool cmu_has_pm)
500 {
501 if (!ctx->auto_clock_gate)
502 return;
503
504 ctx->sysreg = syscon_regmap_lookup_by_phandle(np, "samsung,sysreg");
505 if (IS_ERR(ctx->sysreg)) {
506 pr_warn("%pOF: Unable to get CMU sysreg\n", np);
507 ctx->sysreg = NULL;
508 } else {
509 /* Enable DRCG for all bus components */
510 regmap_write(ctx->sysreg, ctx->drcg_offset, DRCG_EN_MSK);
511 /* Enable memclk gate (not present on all sysreg) */
512 if (ctx->memclk_offset)
513 regmap_write_bits(ctx->sysreg, ctx->memclk_offset,
514 MEMCLK_EN, 0x0);
515
516 if (!cmu_has_pm)
517 /*
518 * When a CMU has PM support, clocks are saved/restored
519 * via its PM handlers, so only register them with the
520 * syscore suspend / resume paths if PM is not in use.
521 */
522 samsung_clk_extended_sleep_init(NULL, ctx->sysreg,
523 cmu->sysreg_clk_regs,
524 cmu->nr_sysreg_clk_regs,
525 NULL, 0);
526 }
527 }
528
529 /*
530 * Common function which registers plls, muxes, dividers and gates
531 * for each CMU. It also add CMU register list to register cache.
532 */
samsung_cmu_register_one(struct device_node * np,const struct samsung_cmu_info * cmu)533 struct samsung_clk_provider * __init samsung_cmu_register_one(
534 struct device_node *np,
535 const struct samsung_cmu_info *cmu)
536 {
537 void __iomem *reg_base;
538 struct samsung_clk_provider *ctx;
539
540 reg_base = of_iomap(np, 0);
541 if (!reg_base) {
542 panic("%s: failed to map registers\n", __func__);
543 return NULL;
544 }
545
546 ctx = samsung_clk_init(NULL, reg_base, cmu->nr_clk_ids);
547 samsung_cmu_register_clocks(ctx, cmu, np);
548
549 if (cmu->clk_regs)
550 samsung_clk_extended_sleep_init(reg_base, NULL,
551 cmu->clk_regs, cmu->nr_clk_regs,
552 cmu->suspend_regs, cmu->nr_suspend_regs);
553
554 samsung_clk_of_add_provider(np, ctx);
555
556 /* sysreg DT nodes reference a clock in this CMU */
557 samsung_en_dyn_root_clk_gating(np, ctx, cmu, false);
558
559 return ctx;
560 }
561