clk-usb.c 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (C) 2018 Microhip / Atmel Corporation
  4. * Wenyou.Yang <wenyou.yang@microchip.com>
  5. */
  6. #include <common.h>
  7. #include <clk-uclass.h>
  8. #include <dm/device.h>
  9. #include <linux/io.h>
  10. #include <mach/at91_pmc.h>
  11. #include "pmc.h"
  12. DECLARE_GLOBAL_DATA_PTR;
  13. #define AT91_USB_CLK_SOURCE_MAX 2
  14. #define AT91_USB_CLK_MAX_DIV 15
  15. struct at91_usb_clk_priv {
  16. u32 num_clksource;
  17. };
  18. static ulong at91_usb_clk_get_rate(struct clk *clk)
  19. {
  20. struct pmc_platdata *plat = dev_get_platdata(clk->dev);
  21. struct at91_pmc *pmc = plat->reg_base;
  22. struct clk source;
  23. u32 tmp, usbdiv;
  24. u8 source_index;
  25. int ret;
  26. tmp = readl(&pmc->pcr);
  27. source_index = (tmp >> AT91_PMC_USB_USBS_OFFSET) &
  28. AT91_PMC_USB_USBS_MASK;
  29. usbdiv = (tmp >> AT91_PMC_USB_DIV_OFFSET) & AT91_PMC_USB_DIV_MASK;
  30. ret = clk_get_by_index(clk->dev, source_index, &source);
  31. if (ret)
  32. return 0;
  33. return clk_get_rate(&source) / (usbdiv + 1);
  34. }
  35. static ulong at91_usb_clk_set_rate(struct clk *clk, ulong rate)
  36. {
  37. struct pmc_platdata *plat = dev_get_platdata(clk->dev);
  38. struct at91_pmc *pmc = plat->reg_base;
  39. struct at91_usb_clk_priv *priv = dev_get_priv(clk->dev);
  40. struct clk source, best_source;
  41. ulong tmp_rate, best_rate = rate, source_rate;
  42. int tmp_diff, best_diff = -1;
  43. u32 div, best_div = 0;
  44. u8 best_source_index = 0;
  45. u8 i;
  46. u32 tmp;
  47. int ret;
  48. for (i = 0; i < priv->num_clksource; i++) {
  49. ret = clk_get_by_index(clk->dev, i, &source);
  50. if (ret)
  51. return ret;
  52. source_rate = clk_get_rate(&source);
  53. if (IS_ERR_VALUE(source_rate))
  54. return source_rate;
  55. for (div = 1; div < AT91_USB_CLK_MAX_DIV + 2; div++) {
  56. tmp_rate = DIV_ROUND_CLOSEST(source_rate, div);
  57. tmp_diff = abs(rate - tmp_rate);
  58. if (best_diff < 0 || best_diff > tmp_diff) {
  59. best_rate = tmp_rate;
  60. best_diff = tmp_diff;
  61. best_div = div - 1;
  62. best_source = source;
  63. best_source_index = i;
  64. }
  65. if (!best_diff || tmp_rate < rate)
  66. break;
  67. }
  68. if (!best_diff)
  69. break;
  70. }
  71. debug("AT91 USB: best sourc: %s, best_rate = %ld, best_div = %d\n",
  72. best_source.dev->name, best_rate, best_div);
  73. ret = clk_enable(&best_source);
  74. if (ret)
  75. return ret;
  76. tmp = AT91_PMC_USB_USBS_(best_source_index) |
  77. AT91_PMC_USB_DIV_(best_div);
  78. writel(tmp, &pmc->usb);
  79. return 0;
  80. }
  81. static struct clk_ops at91_usb_clk_ops = {
  82. .get_rate = at91_usb_clk_get_rate,
  83. .set_rate = at91_usb_clk_set_rate,
  84. };
  85. static int at91_usb_clk_ofdata_to_platdata(struct udevice *dev)
  86. {
  87. struct at91_usb_clk_priv *priv = dev_get_priv(dev);
  88. u32 cells[AT91_USB_CLK_SOURCE_MAX];
  89. u32 num_clksource;
  90. num_clksource = fdtdec_get_int_array_count(gd->fdt_blob,
  91. dev_of_offset(dev),
  92. "clocks", cells,
  93. AT91_USB_CLK_SOURCE_MAX);
  94. if (!num_clksource)
  95. return -1;
  96. priv->num_clksource = num_clksource;
  97. return 0;
  98. }
  99. static int at91_usb_clk_probe(struct udevice *dev)
  100. {
  101. return at91_pmc_core_probe(dev);
  102. }
  103. static const struct udevice_id at91_usb_clk_match[] = {
  104. { .compatible = "atmel,at91sam9x5-clk-usb" },
  105. {}
  106. };
  107. U_BOOT_DRIVER(at91_usb_clk) = {
  108. .name = "at91-usb-clk",
  109. .id = UCLASS_CLK,
  110. .of_match = at91_usb_clk_match,
  111. .probe = at91_usb_clk_probe,
  112. .ofdata_to_platdata = at91_usb_clk_ofdata_to_platdata,
  113. .priv_auto_alloc_size = sizeof(struct at91_usb_clk_priv),
  114. .platdata_auto_alloc_size = sizeof(struct pmc_platdata),
  115. .ops = &at91_usb_clk_ops,
  116. };