1 // SPDX-License-Identifier: GPL-2.0-or-later 2 /* 3 * Copyright (C) 2013 Boris BREZILLON <b.brezillon@overkiz.com> 4 */ 5 6 #include <linux/clk-provider.h> 7 #include <linux/clkdev.h> 8 #include <linux/clk/at91_pmc.h> 9 #include <linux/delay.h> 10 #include <linux/mfd/syscon.h> 11 #include <linux/regmap.h> 12 13 #include "pmc.h" 14 15 #define SLOW_CLOCK_FREQ 32768 16 #define MAINF_DIV 16 17 #define MAINFRDY_TIMEOUT (((MAINF_DIV + 1) * USEC_PER_SEC) / \ 18 SLOW_CLOCK_FREQ) 19 #define MAINF_LOOP_MIN_WAIT (USEC_PER_SEC / SLOW_CLOCK_FREQ) 20 #define MAINF_LOOP_MAX_WAIT MAINFRDY_TIMEOUT 21 22 #define MOR_KEY_MASK (0xff << 16) 23 24 #define clk_main_parent_select(s) (((s) & \ 25 (AT91_PMC_MOSCEN | \ 26 AT91_PMC_OSCBYPASS)) ? 1 : 0) 27 28 struct clk_main_osc { 29 struct clk_hw hw; 30 struct regmap *regmap; 31 struct at91_clk_pms pms; 32 }; 33 34 #define to_clk_main_osc(hw) container_of(hw, struct clk_main_osc, hw) 35 36 struct clk_main_rc_osc { 37 struct clk_hw hw; 38 struct regmap *regmap; 39 unsigned long frequency; 40 unsigned long accuracy; 41 struct at91_clk_pms pms; 42 }; 43 44 #define to_clk_main_rc_osc(hw) container_of(hw, struct clk_main_rc_osc, hw) 45 46 struct clk_rm9200_main { 47 struct clk_hw hw; 48 struct regmap *regmap; 49 }; 50 51 #define to_clk_rm9200_main(hw) container_of(hw, struct clk_rm9200_main, hw) 52 53 struct clk_sam9x5_main { 54 struct clk_hw hw; 55 struct regmap *regmap; 56 struct at91_clk_pms pms; 57 u8 parent; 58 }; 59 60 #define to_clk_sam9x5_main(hw) container_of(hw, struct clk_sam9x5_main, hw) 61 62 static inline bool clk_main_osc_ready(struct regmap *regmap) 63 { 64 unsigned int status; 65 66 regmap_read(regmap, AT91_PMC_SR, &status); 67 68 return status & AT91_PMC_MOSCS; 69 } 70 71 static int clk_main_osc_prepare(struct clk_hw *hw) 72 { 73 struct clk_main_osc *osc = to_clk_main_osc(hw); 74 struct regmap *regmap = osc->regmap; 75 u32 tmp; 76 77 regmap_read(regmap, AT91_CKGR_MOR, &tmp); 78 tmp &= ~MOR_KEY_MASK; 79 80 if (tmp & AT91_PMC_OSCBYPASS) 81 return 0; 82 83 if (!(tmp & AT91_PMC_MOSCEN)) { 84 tmp |= AT91_PMC_MOSCEN | AT91_PMC_KEY; 85 regmap_write(regmap, AT91_CKGR_MOR, tmp); 86 } 87 88 while (!clk_main_osc_ready(regmap)) 89 cpu_relax(); 90 91 return 0; 92 } 93 94 static void clk_main_osc_unprepare(struct clk_hw *hw) 95 { 96 struct clk_main_osc *osc = to_clk_main_osc(hw); 97 struct regmap *regmap = osc->regmap; 98 u32 tmp; 99 100 regmap_read(regmap, AT91_CKGR_MOR, &tmp); 101 if (tmp & AT91_PMC_OSCBYPASS) 102 return; 103 104 if (!(tmp & AT91_PMC_MOSCEN)) 105 return; 106 107 tmp &= ~(AT91_PMC_KEY | AT91_PMC_MOSCEN); 108 regmap_write(regmap, AT91_CKGR_MOR, tmp | AT91_PMC_KEY); 109 } 110 111 static int clk_main_osc_is_prepared(struct clk_hw *hw) 112 { 113 struct clk_main_osc *osc = to_clk_main_osc(hw); 114 struct regmap *regmap = osc->regmap; 115 u32 tmp, status; 116 117 regmap_read(regmap, AT91_CKGR_MOR, &tmp); 118 if (tmp & AT91_PMC_OSCBYPASS) 119 return 1; 120 121 regmap_read(regmap, AT91_PMC_SR, &status); 122 123 return (status & AT91_PMC_MOSCS) && clk_main_parent_select(tmp); 124 } 125 126 static int clk_main_osc_save_context(struct clk_hw *hw) 127 { 128 struct clk_main_osc *osc = to_clk_main_osc(hw); 129 130 osc->pms.status = clk_main_osc_is_prepared(hw); 131 132 return 0; 133 } 134 135 static void clk_main_osc_restore_context(struct clk_hw *hw) 136 { 137 struct clk_main_osc *osc = to_clk_main_osc(hw); 138 139 if (osc->pms.status) 140 clk_main_osc_prepare(hw); 141 } 142 143 static const struct clk_ops main_osc_ops = { 144 .prepare = clk_main_osc_prepare, 145 .unprepare = clk_main_osc_unprepare, 146 .is_prepared = clk_main_osc_is_prepared, 147 .save_context = clk_main_osc_save_context, 148 .restore_context = clk_main_osc_restore_context, 149 }; 150 151 struct clk_hw * __init 152 at91_clk_register_main_osc(struct regmap *regmap, 153 const char *name, 154 const char *parent_name, 155 bool bypass) 156 { 157 struct clk_main_osc *osc; 158 struct clk_init_data init; 159 struct clk_hw *hw; 160 int ret; 161 162 if (!name || !parent_name) 163 return ERR_PTR(-EINVAL); 164 165 osc = kzalloc(sizeof(*osc), GFP_KERNEL); 166 if (!osc) 167 return ERR_PTR(-ENOMEM); 168 169 init.name = name; 170 init.ops = &main_osc_ops; 171 init.parent_names = &parent_name; 172 init.num_parents = 1; 173 init.flags = CLK_IGNORE_UNUSED; 174 175 osc->hw.init = &init; 176 osc->regmap = regmap; 177 178 if (bypass) 179 regmap_update_bits(regmap, 180 AT91_CKGR_MOR, MOR_KEY_MASK | 181 AT91_PMC_OSCBYPASS, 182 AT91_PMC_OSCBYPASS | AT91_PMC_KEY); 183 184 hw = &osc->hw; 185 ret = clk_hw_register(NULL, &osc->hw); 186 if (ret) { 187 kfree(osc); 188 hw = ERR_PTR(ret); 189 } 190 191 return hw; 192 } 193 194 static bool clk_main_rc_osc_ready(struct regmap *regmap) 195 { 196 unsigned int status; 197 198 regmap_read(regmap, AT91_PMC_SR, &status); 199 200 return !!(status & AT91_PMC_MOSCRCS); 201 } 202 203 static int clk_main_rc_osc_prepare(struct clk_hw *hw) 204 { 205 struct clk_main_rc_osc *osc = to_clk_main_rc_osc(hw); 206 struct regmap *regmap = osc->regmap; 207 unsigned int mor; 208 209 regmap_read(regmap, AT91_CKGR_MOR, &mor); 210 211 if (!(mor & AT91_PMC_MOSCRCEN)) 212 regmap_update_bits(regmap, AT91_CKGR_MOR, 213 MOR_KEY_MASK | AT91_PMC_MOSCRCEN, 214 AT91_PMC_MOSCRCEN | AT91_PMC_KEY); 215 216 while (!clk_main_rc_osc_ready(regmap)) 217 cpu_relax(); 218 219 return 0; 220 } 221 222 static void clk_main_rc_osc_unprepare(struct clk_hw *hw) 223 { 224 struct clk_main_rc_osc *osc = to_clk_main_rc_osc(hw); 225 struct regmap *regmap = osc->regmap; 226 unsigned int mor; 227 228 regmap_read(regmap, AT91_CKGR_MOR, &mor); 229 230 if (!(mor & AT91_PMC_MOSCRCEN)) 231 return; 232 233 regmap_update_bits(regmap, AT91_CKGR_MOR, 234 MOR_KEY_MASK | AT91_PMC_MOSCRCEN, AT91_PMC_KEY); 235 } 236 237 static int clk_main_rc_osc_is_prepared(struct clk_hw *hw) 238 { 239 struct clk_main_rc_osc *osc = to_clk_main_rc_osc(hw); 240 struct regmap *regmap = osc->regmap; 241 unsigned int mor, status; 242 243 regmap_read(regmap, AT91_CKGR_MOR, &mor); 244 regmap_read(regmap, AT91_PMC_SR, &status); 245 246 return (mor & AT91_PMC_MOSCRCEN) && (status & AT91_PMC_MOSCRCS); 247 } 248 249 static unsigned long clk_main_rc_osc_recalc_rate(struct clk_hw *hw, 250 unsigned long parent_rate) 251 { 252 struct clk_main_rc_osc *osc = to_clk_main_rc_osc(hw); 253 254 return osc->frequency; 255 } 256 257 static unsigned long clk_main_rc_osc_recalc_accuracy(struct clk_hw *hw, 258 unsigned long parent_acc) 259 { 260 struct clk_main_rc_osc *osc = to_clk_main_rc_osc(hw); 261 262 return osc->accuracy; 263 } 264 265 static int clk_main_rc_osc_save_context(struct clk_hw *hw) 266 { 267 struct clk_main_rc_osc *osc = to_clk_main_rc_osc(hw); 268 269 osc->pms.status = clk_main_rc_osc_is_prepared(hw); 270 271 return 0; 272 } 273 274 static void clk_main_rc_osc_restore_context(struct clk_hw *hw) 275 { 276 struct clk_main_rc_osc *osc = to_clk_main_rc_osc(hw); 277 278 if (osc->pms.status) 279 clk_main_rc_osc_prepare(hw); 280 } 281 282 static const struct clk_ops main_rc_osc_ops = { 283 .prepare = clk_main_rc_osc_prepare, 284 .unprepare = clk_main_rc_osc_unprepare, 285 .is_prepared = clk_main_rc_osc_is_prepared, 286 .recalc_rate = clk_main_rc_osc_recalc_rate, 287 .recalc_accuracy = clk_main_rc_osc_recalc_accuracy, 288 .save_context = clk_main_rc_osc_save_context, 289 .restore_context = clk_main_rc_osc_restore_context, 290 }; 291 292 struct clk_hw * __init 293 at91_clk_register_main_rc_osc(struct regmap *regmap, 294 const char *name, 295 u32 frequency, u32 accuracy) 296 { 297 struct clk_main_rc_osc *osc; 298 struct clk_init_data init; 299 struct clk_hw *hw; 300 int ret; 301 302 if (!name || !frequency) 303 return ERR_PTR(-EINVAL); 304 305 osc = kzalloc(sizeof(*osc), GFP_KERNEL); 306 if (!osc) 307 return ERR_PTR(-ENOMEM); 308 309 init.name = name; 310 init.ops = &main_rc_osc_ops; 311 init.parent_names = NULL; 312 init.num_parents = 0; 313 init.flags = CLK_IGNORE_UNUSED; 314 315 osc->hw.init = &init; 316 osc->regmap = regmap; 317 osc->frequency = frequency; 318 osc->accuracy = accuracy; 319 320 hw = &osc->hw; 321 ret = clk_hw_register(NULL, hw); 322 if (ret) { 323 kfree(osc); 324 hw = ERR_PTR(ret); 325 } 326 327 return hw; 328 } 329 330 static int clk_main_probe_frequency(struct regmap *regmap) 331 { 332 unsigned long prep_time, timeout; 333 unsigned int mcfr; 334 335 timeout = jiffies + usecs_to_jiffies(MAINFRDY_TIMEOUT); 336 do { 337 prep_time = jiffies; 338 regmap_read(regmap, AT91_CKGR_MCFR, &mcfr); 339 if (mcfr & AT91_PMC_MAINRDY) 340 return 0; 341 if (system_state < SYSTEM_RUNNING) 342 udelay(MAINF_LOOP_MIN_WAIT); 343 else 344 usleep_range(MAINF_LOOP_MIN_WAIT, MAINF_LOOP_MAX_WAIT); 345 } while (time_before(prep_time, timeout)); 346 347 return -ETIMEDOUT; 348 } 349 350 static unsigned long clk_main_recalc_rate(struct regmap *regmap, 351 unsigned long parent_rate) 352 { 353 unsigned int mcfr; 354 355 if (parent_rate) 356 return parent_rate; 357 358 pr_warn("Main crystal frequency not set, using approximate value\n"); 359 regmap_read(regmap, AT91_CKGR_MCFR, &mcfr); 360 if (!(mcfr & AT91_PMC_MAINRDY)) 361 return 0; 362 363 return ((mcfr & AT91_PMC_MAINF) * SLOW_CLOCK_FREQ) / MAINF_DIV; 364 } 365 366 static int clk_rm9200_main_prepare(struct clk_hw *hw) 367 { 368 struct clk_rm9200_main *clkmain = to_clk_rm9200_main(hw); 369 370 return clk_main_probe_frequency(clkmain->regmap); 371 } 372 373 static int clk_rm9200_main_is_prepared(struct clk_hw *hw) 374 { 375 struct clk_rm9200_main *clkmain = to_clk_rm9200_main(hw); 376 unsigned int status; 377 378 regmap_read(clkmain->regmap, AT91_CKGR_MCFR, &status); 379 380 return !!(status & AT91_PMC_MAINRDY); 381 } 382 383 static unsigned long clk_rm9200_main_recalc_rate(struct clk_hw *hw, 384 unsigned long parent_rate) 385 { 386 struct clk_rm9200_main *clkmain = to_clk_rm9200_main(hw); 387 388 return clk_main_recalc_rate(clkmain->regmap, parent_rate); 389 } 390 391 static const struct clk_ops rm9200_main_ops = { 392 .prepare = clk_rm9200_main_prepare, 393 .is_prepared = clk_rm9200_main_is_prepared, 394 .recalc_rate = clk_rm9200_main_recalc_rate, 395 }; 396 397 struct clk_hw * __init 398 at91_clk_register_rm9200_main(struct regmap *regmap, 399 const char *name, 400 const char *parent_name) 401 { 402 struct clk_rm9200_main *clkmain; 403 struct clk_init_data init; 404 struct clk_hw *hw; 405 int ret; 406 407 if (!name) 408 return ERR_PTR(-EINVAL); 409 410 if (!parent_name) 411 return ERR_PTR(-EINVAL); 412 413 clkmain = kzalloc(sizeof(*clkmain), GFP_KERNEL); 414 if (!clkmain) 415 return ERR_PTR(-ENOMEM); 416 417 init.name = name; 418 init.ops = &rm9200_main_ops; 419 init.parent_names = &parent_name; 420 init.num_parents = 1; 421 init.flags = 0; 422 423 clkmain->hw.init = &init; 424 clkmain->regmap = regmap; 425 426 hw = &clkmain->hw; 427 ret = clk_hw_register(NULL, &clkmain->hw); 428 if (ret) { 429 kfree(clkmain); 430 hw = ERR_PTR(ret); 431 } 432 433 return hw; 434 } 435 436 static inline bool clk_sam9x5_main_ready(struct regmap *regmap) 437 { 438 unsigned int status; 439 440 regmap_read(regmap, AT91_PMC_SR, &status); 441 442 return !!(status & AT91_PMC_MOSCSELS); 443 } 444 445 static int clk_sam9x5_main_prepare(struct clk_hw *hw) 446 { 447 struct clk_sam9x5_main *clkmain = to_clk_sam9x5_main(hw); 448 struct regmap *regmap = clkmain->regmap; 449 450 while (!clk_sam9x5_main_ready(regmap)) 451 cpu_relax(); 452 453 return clk_main_probe_frequency(regmap); 454 } 455 456 static int clk_sam9x5_main_is_prepared(struct clk_hw *hw) 457 { 458 struct clk_sam9x5_main *clkmain = to_clk_sam9x5_main(hw); 459 460 return clk_sam9x5_main_ready(clkmain->regmap); 461 } 462 463 static unsigned long clk_sam9x5_main_recalc_rate(struct clk_hw *hw, 464 unsigned long parent_rate) 465 { 466 struct clk_sam9x5_main *clkmain = to_clk_sam9x5_main(hw); 467 468 return clk_main_recalc_rate(clkmain->regmap, parent_rate); 469 } 470 471 static int clk_sam9x5_main_set_parent(struct clk_hw *hw, u8 index) 472 { 473 struct clk_sam9x5_main *clkmain = to_clk_sam9x5_main(hw); 474 struct regmap *regmap = clkmain->regmap; 475 unsigned int tmp; 476 477 if (index > 1) 478 return -EINVAL; 479 480 regmap_read(regmap, AT91_CKGR_MOR, &tmp); 481 482 if (index && !(tmp & AT91_PMC_MOSCSEL)) 483 tmp = AT91_PMC_MOSCSEL; 484 else if (!index && (tmp & AT91_PMC_MOSCSEL)) 485 tmp = 0; 486 else 487 return 0; 488 489 regmap_update_bits(regmap, AT91_CKGR_MOR, 490 AT91_PMC_MOSCSEL | MOR_KEY_MASK, 491 tmp | AT91_PMC_KEY); 492 493 while (!clk_sam9x5_main_ready(regmap)) 494 cpu_relax(); 495 496 return 0; 497 } 498 499 static u8 clk_sam9x5_main_get_parent(struct clk_hw *hw) 500 { 501 struct clk_sam9x5_main *clkmain = to_clk_sam9x5_main(hw); 502 unsigned int status; 503 504 regmap_read(clkmain->regmap, AT91_CKGR_MOR, &status); 505 506 return clk_main_parent_select(status); 507 } 508 509 static int clk_sam9x5_main_save_context(struct clk_hw *hw) 510 { 511 struct clk_sam9x5_main *clkmain = to_clk_sam9x5_main(hw); 512 513 clkmain->pms.status = clk_main_rc_osc_is_prepared(&clkmain->hw); 514 clkmain->pms.parent = clk_sam9x5_main_get_parent(&clkmain->hw); 515 516 return 0; 517 } 518 519 static void clk_sam9x5_main_restore_context(struct clk_hw *hw) 520 { 521 struct clk_sam9x5_main *clkmain = to_clk_sam9x5_main(hw); 522 int ret; 523 524 ret = clk_sam9x5_main_set_parent(hw, clkmain->pms.parent); 525 if (ret) 526 return; 527 528 if (clkmain->pms.status) 529 clk_sam9x5_main_prepare(hw); 530 } 531 532 static const struct clk_ops sam9x5_main_ops = { 533 .prepare = clk_sam9x5_main_prepare, 534 .is_prepared = clk_sam9x5_main_is_prepared, 535 .recalc_rate = clk_sam9x5_main_recalc_rate, 536 .determine_rate = clk_hw_determine_rate_no_reparent, 537 .set_parent = clk_sam9x5_main_set_parent, 538 .get_parent = clk_sam9x5_main_get_parent, 539 .save_context = clk_sam9x5_main_save_context, 540 .restore_context = clk_sam9x5_main_restore_context, 541 }; 542 543 struct clk_hw * __init 544 at91_clk_register_sam9x5_main(struct regmap *regmap, 545 const char *name, 546 const char **parent_names, 547 int num_parents) 548 { 549 struct clk_sam9x5_main *clkmain; 550 struct clk_init_data init; 551 unsigned int status; 552 struct clk_hw *hw; 553 int ret; 554 555 if (!name) 556 return ERR_PTR(-EINVAL); 557 558 if (!parent_names || !num_parents) 559 return ERR_PTR(-EINVAL); 560 561 clkmain = kzalloc(sizeof(*clkmain), GFP_KERNEL); 562 if (!clkmain) 563 return ERR_PTR(-ENOMEM); 564 565 init.name = name; 566 init.ops = &sam9x5_main_ops; 567 init.parent_names = parent_names; 568 init.num_parents = num_parents; 569 init.flags = CLK_SET_PARENT_GATE; 570 571 clkmain->hw.init = &init; 572 clkmain->regmap = regmap; 573 regmap_read(clkmain->regmap, AT91_CKGR_MOR, &status); 574 clkmain->parent = clk_main_parent_select(status); 575 576 hw = &clkmain->hw; 577 ret = clk_hw_register(NULL, &clkmain->hw); 578 if (ret) { 579 kfree(clkmain); 580 hw = ERR_PTR(ret); 581 } 582 583 return hw; 584 } 585