imx-gpt-timer.c 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (C) 2021
  4. * Author(s): Giulio Benetti <giulio.benetti@benettiengineering.com>
  5. */
  6. #include <common.h>
  7. #include <clk.h>
  8. #include <dm.h>
  9. #include <fdtdec.h>
  10. #include <timer.h>
  11. #include <dm/device_compat.h>
  12. #include <asm/io.h>
  13. #define GPT_CR_EN BIT(0)
  14. #define GPT_CR_FRR BIT(9)
  15. #define GPT_CR_EN_24M BIT(10)
  16. #define GPT_CR_SWR BIT(15)
  17. #define GPT_PR_PRESCALER24M_MASK 0x0000F000
  18. #define GPT_PR_PRESCALER24M_SHIFT 12
  19. #define GPT_PR_PRESCALER24M_MAX (GPT_PR_PRESCALER24M_MASK >> GPT_PR_PRESCALER24M_SHIFT)
  20. #define GPT_PR_PRESCALER_MASK 0x00000FFF
  21. #define GPT_PR_PRESCALER_SHIFT 0
  22. #define GPT_PR_PRESCALER_MAX (GPT_PR_PRESCALER_MASK >> GPT_PR_PRESCALER_SHIFT)
  23. #define GPT_CLKSRC_IPG_CLK (1 << 6)
  24. #define GPT_CLKSRC_IPG_CLK_24M (5 << 6)
  25. /* If CFG_SYS_HZ_CLOCK not specified et's default to 3Mhz */
  26. #ifndef CFG_SYS_HZ_CLOCK
  27. #define CFG_SYS_HZ_CLOCK 3000000
  28. #endif
  29. struct imx_gpt_timer_regs {
  30. u32 cr;
  31. u32 pr;
  32. u32 sr;
  33. u32 ir;
  34. u32 ocr1;
  35. u32 ocr2;
  36. u32 ocr3;
  37. u32 icr1;
  38. u32 icr2;
  39. u32 cnt;
  40. };
  41. struct imx_gpt_timer_priv {
  42. struct imx_gpt_timer_regs *base;
  43. };
  44. static u64 imx_gpt_timer_get_count(struct udevice *dev)
  45. {
  46. struct imx_gpt_timer_priv *priv = dev_get_priv(dev);
  47. struct imx_gpt_timer_regs *regs = priv->base;
  48. return timer_conv_64(readl(&regs->cnt));
  49. }
  50. static int imx_gpt_setup(struct imx_gpt_timer_regs *regs, u32 rate)
  51. {
  52. u32 prescaler = (rate / CFG_SYS_HZ_CLOCK) - 1;
  53. /* Reset the timer */
  54. setbits_le32(&regs->cr, GPT_CR_SWR);
  55. /* Wait for timer to finish reset */
  56. while (readl(&regs->cr) & GPT_CR_SWR)
  57. ;
  58. if (rate == 24000000UL) {
  59. /* Set timer frequency if using 24M clock source */
  60. if (prescaler > GPT_PR_PRESCALER24M_MAX)
  61. return -EINVAL;
  62. /* Set 24M prescaler */
  63. writel((prescaler << GPT_PR_PRESCALER24M_SHIFT), &regs->pr);
  64. /* Set Oscillator as clock source, enable 24M input and set gpt
  65. * in free-running mode
  66. */
  67. writel(GPT_CLKSRC_IPG_CLK_24M | GPT_CR_EN_24M | GPT_CR_FRR, &regs->cr);
  68. } else {
  69. if (prescaler > GPT_PR_PRESCALER_MAX)
  70. return -EINVAL;
  71. /* Set prescaler */
  72. writel((prescaler << GPT_PR_PRESCALER_SHIFT), &regs->pr);
  73. /* Set Peripheral as clock source and set gpt in free-running
  74. * mode
  75. */
  76. writel(GPT_CLKSRC_IPG_CLK | GPT_CR_FRR, &regs->cr);
  77. }
  78. /* Start timer */
  79. setbits_le32(&regs->cr, GPT_CR_EN);
  80. return 0;
  81. }
  82. static int imx_gpt_timer_probe(struct udevice *dev)
  83. {
  84. struct timer_dev_priv *uc_priv = dev_get_uclass_priv(dev);
  85. struct imx_gpt_timer_priv *priv = dev_get_priv(dev);
  86. struct imx_gpt_timer_regs *regs;
  87. struct clk clk;
  88. fdt_addr_t addr;
  89. u32 clk_rate;
  90. int ret;
  91. addr = dev_read_addr(dev);
  92. if (addr == FDT_ADDR_T_NONE)
  93. return -EINVAL;
  94. priv->base = (struct imx_gpt_timer_regs *)addr;
  95. regs = priv->base;
  96. ret = clk_get_by_index(dev, 0, &clk);
  97. if (ret < 0)
  98. return ret;
  99. ret = clk_enable(&clk);
  100. if (ret) {
  101. dev_err(dev, "Failed to enable clock\n");
  102. return ret;
  103. }
  104. /* Get timer clock rate */
  105. clk_rate = clk_get_rate(&clk);
  106. if (clk_rate <= 0) {
  107. dev_err(dev, "Could not get clock rate...\n");
  108. return -EINVAL;
  109. }
  110. ret = imx_gpt_setup(regs, clk_rate);
  111. if (ret) {
  112. dev_err(dev, "Could not setup timer\n");
  113. return ret;
  114. }
  115. uc_priv->clock_rate = CFG_SYS_HZ_CLOCK;
  116. return 0;
  117. }
  118. static const struct timer_ops imx_gpt_timer_ops = {
  119. .get_count = imx_gpt_timer_get_count,
  120. };
  121. static const struct udevice_id imx_gpt_timer_ids[] = {
  122. { .compatible = "fsl,imxrt-gpt" },
  123. {}
  124. };
  125. U_BOOT_DRIVER(imx_gpt_timer) = {
  126. .name = "imx_gpt_timer",
  127. .id = UCLASS_TIMER,
  128. .of_match = imx_gpt_timer_ids,
  129. .priv_auto = sizeof(struct imx_gpt_timer_priv),
  130. .probe = imx_gpt_timer_probe,
  131. .ops = &imx_gpt_timer_ops,
  132. };