timer.c 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  1. /*
  2. * (C) Copyright 2011
  3. * Marvell Semiconductor <www.marvell.com>
  4. * Written-by: Lei Wen <leiwen@marvell.com>
  5. *
  6. * SPDX-License-Identifier: GPL-2.0+
  7. */
  8. #include <common.h>
  9. #include <asm/arch/cpu.h>
  10. #include <asm/arch/pantheon.h>
  11. /*
  12. * Timer registers
  13. * Refer 6.2.9 in Datasheet
  14. */
  15. struct panthtmr_registers {
  16. u32 clk_ctrl; /* Timer clk control reg */
  17. u32 match[9]; /* Timer match registers */
  18. u32 count[3]; /* Timer count registers */
  19. u32 status[3];
  20. u32 ie[3];
  21. u32 preload[3]; /* Timer preload value */
  22. u32 preload_ctrl[3];
  23. u32 wdt_match_en;
  24. u32 wdt_match_r;
  25. u32 wdt_val;
  26. u32 wdt_sts;
  27. u32 icr[3];
  28. u32 wdt_icr;
  29. u32 cer; /* Timer count enable reg */
  30. u32 cmr;
  31. u32 ilr[3];
  32. u32 wcr;
  33. u32 wfar;
  34. u32 wsar;
  35. u32 cvwr[3];
  36. };
  37. #define TIMER 0 /* Use TIMER 0 */
  38. /* Each timer has 3 match registers */
  39. #define MATCH_CMP(x) ((3 * TIMER) + x)
  40. #define TIMER_LOAD_VAL 0xffffffff
  41. #define COUNT_RD_REQ 0x1
  42. DECLARE_GLOBAL_DATA_PTR;
  43. /* Using gd->arch.tbu from timestamp and gd->arch.tbl for lastdec */
  44. /*
  45. * For preventing risk of instability in reading counter value,
  46. * first set read request to register cvwr and then read same
  47. * register after it captures counter value.
  48. */
  49. ulong read_timer(void)
  50. {
  51. struct panthtmr_registers *panthtimers =
  52. (struct panthtmr_registers *) PANTHEON_TIMER_BASE;
  53. volatile int loop=100;
  54. ulong val;
  55. writel(COUNT_RD_REQ, &panthtimers->cvwr);
  56. while (loop--)
  57. val = readl(&panthtimers->cvwr);
  58. /*
  59. * This stop gcc complain and prevent loop mistake init to 0
  60. */
  61. val = readl(&panthtimers->cvwr);
  62. return val;
  63. }
  64. ulong get_timer_masked(void)
  65. {
  66. ulong now = read_timer();
  67. if (now >= gd->arch.tbl) {
  68. /* normal mode */
  69. gd->arch.tbu += now - gd->arch.tbl;
  70. } else {
  71. /* we have an overflow ... */
  72. gd->arch.tbu += now + TIMER_LOAD_VAL - gd->arch.tbl;
  73. }
  74. gd->arch.tbl = now;
  75. return gd->arch.tbu;
  76. }
  77. ulong get_timer(ulong base)
  78. {
  79. return ((get_timer_masked() / (CONFIG_SYS_HZ_CLOCK / 1000)) -
  80. base);
  81. }
  82. void __udelay(unsigned long usec)
  83. {
  84. ulong delayticks;
  85. ulong endtime;
  86. delayticks = (usec * (CONFIG_SYS_HZ_CLOCK / 1000000));
  87. endtime = get_timer_masked() + delayticks;
  88. while (get_timer_masked() < endtime)
  89. ;
  90. }
  91. /*
  92. * init the Timer
  93. */
  94. int timer_init(void)
  95. {
  96. struct panthapb_registers *apb1clkres =
  97. (struct panthapb_registers *) PANTHEON_APBC_BASE;
  98. struct panthtmr_registers *panthtimers =
  99. (struct panthtmr_registers *) PANTHEON_TIMER_BASE;
  100. /* Enable Timer clock at 3.25 MHZ */
  101. writel(APBC_APBCLK | APBC_FNCLK | APBC_FNCLKSEL(3), &apb1clkres->timers);
  102. /* load value into timer */
  103. writel(0x0, &panthtimers->clk_ctrl);
  104. /* Use Timer 0 Match Resiger 0 */
  105. writel(TIMER_LOAD_VAL, &panthtimers->match[MATCH_CMP(0)]);
  106. /* Preload value is 0 */
  107. writel(0x0, &panthtimers->preload[TIMER]);
  108. /* Enable match comparator 0 for Timer 0 */
  109. writel(0x1, &panthtimers->preload_ctrl[TIMER]);
  110. /* Enable timer 0 */
  111. writel(0x1, &panthtimers->cer);
  112. /* init the gd->arch.tbu and gd->arch.tbl value */
  113. gd->arch.tbl = read_timer();
  114. gd->arch.tbu = 0;
  115. return 0;
  116. }
  117. #define MPMU_APRR_WDTR (1<<4)
  118. #define TMR_WFAR 0xbaba /* WDT Register First key */
  119. #define TMP_WSAR 0xeb10 /* WDT Register Second key */
  120. /*
  121. * This function uses internal Watchdog Timer
  122. * based reset mechanism.
  123. * Steps to write watchdog registers (protected access)
  124. * 1. Write key value to TMR_WFAR reg.
  125. * 2. Write key value to TMP_WSAR reg.
  126. * 3. Perform write operation.
  127. */
  128. void reset_cpu (unsigned long ignored)
  129. {
  130. struct panthmpmu_registers *mpmu =
  131. (struct panthmpmu_registers *) PANTHEON_MPMU_BASE;
  132. struct panthtmr_registers *panthtimers =
  133. (struct panthtmr_registers *) PANTHEON_WD_TIMER_BASE;
  134. u32 val;
  135. /* negate hardware reset to the WDT after system reset */
  136. val = readl(&mpmu->aprr);
  137. val = val | MPMU_APRR_WDTR;
  138. writel(val, &mpmu->aprr);
  139. /* reset/enable WDT clock */
  140. writel(APBC_APBCLK, &mpmu->wdtpcr);
  141. /* clear previous WDT status */
  142. writel(TMR_WFAR, &panthtimers->wfar);
  143. writel(TMP_WSAR, &panthtimers->wsar);
  144. writel(0, &panthtimers->wdt_sts);
  145. /* set match counter */
  146. writel(TMR_WFAR, &panthtimers->wfar);
  147. writel(TMP_WSAR, &panthtimers->wsar);
  148. writel(0xf, &panthtimers->wdt_match_r);
  149. /* enable WDT reset */
  150. writel(TMR_WFAR, &panthtimers->wfar);
  151. writel(TMP_WSAR, &panthtimers->wsar);
  152. writel(0x3, &panthtimers->wdt_match_en);
  153. /*enable functional WDT clock */
  154. writel(APBC_APBCLK | APBC_FNCLK, &mpmu->wdtpcr);
  155. }
  156. /*
  157. * This function is derived from PowerPC code (read timebase as long long).
  158. * On ARM it just returns the timer value.
  159. */
  160. unsigned long long get_ticks(void)
  161. {
  162. return get_timer(0);
  163. }
  164. /*
  165. * This function is derived from PowerPC code (timebase clock frequency).
  166. * On ARM it returns the number of timer ticks per second.
  167. */
  168. ulong get_tbclk (void)
  169. {
  170. return (ulong)CONFIG_SYS_HZ;
  171. }