clk-bcm53573-ilp.c 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (C) 2016 Rafał Miłecki <[email protected]>
  4. */
  5. #include <linux/clk-provider.h>
  6. #include <linux/err.h>
  7. #include <linux/io.h>
  8. #include <linux/mfd/syscon.h>
  9. #include <linux/of.h>
  10. #include <linux/of_address.h>
  11. #include <linux/regmap.h>
  12. #include <linux/slab.h>
  13. #define PMU_XTAL_FREQ_RATIO 0x66c
  14. #define XTAL_ALP_PER_4ILP 0x00001fff
  15. #define XTAL_CTL_EN 0x80000000
  16. #define PMU_SLOW_CLK_PERIOD 0x6dc
  17. struct bcm53573_ilp {
  18. struct clk_hw hw;
  19. struct regmap *regmap;
  20. };
  21. static int bcm53573_ilp_enable(struct clk_hw *hw)
  22. {
  23. struct bcm53573_ilp *ilp = container_of(hw, struct bcm53573_ilp, hw);
  24. regmap_write(ilp->regmap, PMU_SLOW_CLK_PERIOD, 0x10199);
  25. regmap_write(ilp->regmap, 0x674, 0x10000);
  26. return 0;
  27. }
  28. static void bcm53573_ilp_disable(struct clk_hw *hw)
  29. {
  30. struct bcm53573_ilp *ilp = container_of(hw, struct bcm53573_ilp, hw);
  31. regmap_write(ilp->regmap, PMU_SLOW_CLK_PERIOD, 0);
  32. regmap_write(ilp->regmap, 0x674, 0);
  33. }
  34. static unsigned long bcm53573_ilp_recalc_rate(struct clk_hw *hw,
  35. unsigned long parent_rate)
  36. {
  37. struct bcm53573_ilp *ilp = container_of(hw, struct bcm53573_ilp, hw);
  38. struct regmap *regmap = ilp->regmap;
  39. u32 last_val, cur_val;
  40. int sum = 0, num = 0, loop_num = 0;
  41. int avg;
  42. /* Enable measurement */
  43. regmap_write(regmap, PMU_XTAL_FREQ_RATIO, XTAL_CTL_EN);
  44. /* Read initial value */
  45. regmap_read(regmap, PMU_XTAL_FREQ_RATIO, &last_val);
  46. last_val &= XTAL_ALP_PER_4ILP;
  47. /*
  48. * At minimum we should loop for a bit to let hardware do the
  49. * measurement. This isn't very accurate however, so for a better
  50. * precision lets try getting 20 different values for and use average.
  51. */
  52. while (num < 20) {
  53. regmap_read(regmap, PMU_XTAL_FREQ_RATIO, &cur_val);
  54. cur_val &= XTAL_ALP_PER_4ILP;
  55. if (cur_val != last_val) {
  56. /* Got different value, use it */
  57. sum += cur_val;
  58. num++;
  59. loop_num = 0;
  60. last_val = cur_val;
  61. } else if (++loop_num > 5000) {
  62. /* Same value over and over, give up */
  63. sum += cur_val;
  64. num++;
  65. break;
  66. }
  67. cpu_relax();
  68. }
  69. /* Disable measurement to save power */
  70. regmap_write(regmap, PMU_XTAL_FREQ_RATIO, 0x0);
  71. avg = sum / num;
  72. return parent_rate * 4 / avg;
  73. }
  74. static const struct clk_ops bcm53573_ilp_clk_ops = {
  75. .enable = bcm53573_ilp_enable,
  76. .disable = bcm53573_ilp_disable,
  77. .recalc_rate = bcm53573_ilp_recalc_rate,
  78. };
  79. static void bcm53573_ilp_init(struct device_node *np)
  80. {
  81. struct bcm53573_ilp *ilp;
  82. struct clk_init_data init = { };
  83. const char *parent_name;
  84. int err;
  85. ilp = kzalloc(sizeof(*ilp), GFP_KERNEL);
  86. if (!ilp)
  87. return;
  88. parent_name = of_clk_get_parent_name(np, 0);
  89. if (!parent_name) {
  90. err = -ENOENT;
  91. goto err_free_ilp;
  92. }
  93. ilp->regmap = syscon_node_to_regmap(of_get_parent(np));
  94. if (IS_ERR(ilp->regmap)) {
  95. err = PTR_ERR(ilp->regmap);
  96. goto err_free_ilp;
  97. }
  98. init.name = np->name;
  99. init.ops = &bcm53573_ilp_clk_ops;
  100. init.parent_names = &parent_name;
  101. init.num_parents = 1;
  102. ilp->hw.init = &init;
  103. err = clk_hw_register(NULL, &ilp->hw);
  104. if (err)
  105. goto err_free_ilp;
  106. err = of_clk_add_hw_provider(np, of_clk_hw_simple_get, &ilp->hw);
  107. if (err)
  108. goto err_clk_hw_unregister;
  109. return;
  110. err_clk_hw_unregister:
  111. clk_hw_unregister(&ilp->hw);
  112. err_free_ilp:
  113. kfree(ilp);
  114. pr_err("Failed to init ILP clock: %d\n", err);
  115. }
  116. /* We need it very early for arch code, before device model gets ready */
  117. CLK_OF_DECLARE(bcm53573_ilp_clk, "brcm,bcm53573-ilp", bcm53573_ilp_init);