acts_timer.c 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. /*
  2. * Copyright (c) 2017 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <kernel.h>
  7. #include <drivers/timer/system_timer.h>
  8. #include <sys_clock.h>
  9. #include <spinlock.h>
  10. #include <soc.h>
  11. #define USE_T2_FOR_CYCLE
  12. #define CYC_PER_TICK (sys_clock_hw_cycles_per_sec() \
  13. / CONFIG_SYS_CLOCK_TICKS_PER_SEC)
  14. #define MAX_TICKS ((0x40000000 / CYC_PER_TICK) - 1)
  15. #define MAX_CYCLES (MAX_TICKS * CYC_PER_TICK)
  16. static struct k_spinlock lock;
  17. /*
  18. * This local variable holds the amount of SysTick HW cycles elapsed
  19. * and it is updated in z_clock_isr() and sys_clock_set_timeout().
  20. *
  21. * Note:
  22. * At an arbitrary point in time the "current" value of the SysTick
  23. * HW timer is calculated as:
  24. *
  25. * t = cycle_counter + elapsed();
  26. */
  27. static uint32_t cycle_count;
  28. /*
  29. * This local variable holds the amount of elapsed SysTick HW cycles
  30. * that have been announced to the kernel.
  31. */
  32. static uint32_t announced_cycles;
  33. /**/
  34. static uint32_t last_cycle_count;
  35. static uint32_t sleep_cycle_count;
  36. #if defined(CONFIG_PM_DEVICE)
  37. static uint32_t sleep_st_cycle;
  38. #endif
  39. static uint32_t acts_timer_elapsed(void)
  40. {
  41. return sys_read32(T0_CNT)-last_cycle_count;
  42. }
  43. #ifdef USE_T2_FOR_CYCLE
  44. static int timer_cyc_init(const struct device *arg)
  45. {
  46. #ifdef CONFIG_SOC_SERIES_LEOPARD
  47. acts_clock_peripheral_enable(CLOCK_ID_TIMER2);
  48. acts_reset_peripheral(CLOCK_ID_TIMER2);
  49. #endif
  50. sys_write32(0x0, CMU_TIMER2CLK); //select hosc
  51. sys_write32(0x1, T2_CTL); //clear pending stop timer
  52. timer_reg_wait();
  53. sys_write32(TIMER_MAX_CYCLES_VALUE, T2_VAL);
  54. sys_write32(0x824, T2_CTL); /* enable counter up, reload, notirq */
  55. }
  56. /*early init for use k_busy_wait*/
  57. SYS_INIT(timer_cyc_init, PRE_KERNEL_1, 0);
  58. #endif
  59. static void acts_timer_set_next_irq(uint32_t cycle)
  60. {
  61. if(cycle < CYC_PER_TICK/100) //min: 10us
  62. cycle = CYC_PER_TICK/100;
  63. if(cycle > MAX_CYCLES)
  64. cycle = MAX_CYCLES;
  65. sys_write32(sys_read32(T0_CNT)+cycle, T0_VAL);
  66. }
  67. /* Callout out of platform assembly, not hooked via IRQ_CONNECT... */
  68. void acts_timer_isr(void *arg)
  69. {
  70. ARG_UNUSED(arg);
  71. uint32_t dticks, cycle;
  72. /* Increment the amount of HW cycles elapsed (complete counter
  73. * cycles) and announce the progress to the kernel.
  74. */
  75. sys_write32(sys_read32(T0_CTL), T0_CTL); // clear penddig;
  76. cycle = sys_read32(T0_CNT);
  77. cycle_count += cycle-last_cycle_count;
  78. last_cycle_count = cycle;
  79. if(sleep_cycle_count){
  80. cycle_count += sleep_cycle_count;
  81. sleep_cycle_count = 0;
  82. }
  83. #if defined(CONFIG_TICKLESS_KERNEL)
  84. dticks = (cycle_count - announced_cycles) / CYC_PER_TICK;
  85. announced_cycles += dticks * CYC_PER_TICK;
  86. //acts_timer_set_next_irq(sys_clock_hw_cycles_per_sec()*10); //Guarantee 10 seconds of interruption
  87. sys_clock_announce(dticks);
  88. #else
  89. acts_timer_set_next_irq(CYC_PER_TICK);
  90. sys_clock_announce(1);
  91. #endif
  92. //printk("-ie=0x%x, 0x%x, 0x%x\n", cycle_count, sys_read32(T0_VAL), sys_read32(T0_CNT));
  93. }
  94. int sys_clock_driver_init(const struct device *device)
  95. {
  96. /* init timer0 as clock event device */
  97. sys_write32(0x0, CMU_TIMER0CLK); //select hosc
  98. #ifdef CONFIG_SOC_SERIES_LEOPARD
  99. acts_clock_peripheral_enable(CLOCK_ID_TIMER0);
  100. #else
  101. acts_clock_peripheral_enable(CLOCK_ID_TIMER);
  102. #endif
  103. sys_write32(0x1, T0_CTL); //clear pending stop timer
  104. timer_reg_wait();
  105. sys_write32(0xe22, T0_CTL); /* enable counter up, continue mode, irq */
  106. acts_timer_set_next_irq(CYC_PER_TICK);
  107. //printk("sys_clock_driver_init, cnt=0x%x, val=0x%x\n", sys_read32(T0_CNT), sys_read32(T0_VAL));
  108. IRQ_CONNECT(IRQ_ID_TIMER0, 1, acts_timer_isr, 0, 0);
  109. irq_enable(IRQ_ID_TIMER0);
  110. return 0;
  111. }
  112. #define RC32K_MUTIPLE soc_rc32K_mutiple_hosc()
  113. void sys_clock_set_timeout(int32_t ticks, bool idle)
  114. {
  115. #if defined(CONFIG_TICKLESS_KERNEL)
  116. k_spinlock_key_t key = k_spin_lock(&lock);
  117. uint32_t delay;
  118. ticks = (ticks == K_TICKS_FOREVER) ? MAX_TICKS : ticks;
  119. if(!ticks)
  120. ticks = 1;
  121. uint32_t unannounced = cycle_count - announced_cycles + acts_timer_elapsed();
  122. /* Desired delay in the future */
  123. delay = ticks * CYC_PER_TICK;
  124. if(unannounced > delay){
  125. acts_timer_set_next_irq(20); // now is irq
  126. }else{
  127. delay = delay - unannounced + 1;
  128. acts_timer_set_next_irq(delay);
  129. }
  130. k_spin_unlock(&lock, key);
  131. #endif
  132. }
  133. #if defined(CONFIG_PM_DEVICE)
  134. #include <wait_q.h>
  135. int sys_clock_device_ctrl(const struct device *device, enum pm_device_action action)
  136. {
  137. if(action == PM_DEVICE_ACTION_RESUME){
  138. uint32_t cycle = 0;
  139. if(sleep_st_cycle){
  140. cycle = sys_read32(T0_CNT);
  141. sys_write32(0x0, CMU_TIMER0CLK); //select hosc
  142. cycle -= sleep_st_cycle;
  143. sleep_cycle_count = cycle*RC32K_MUTIPLE;
  144. sleep_st_cycle = 0;
  145. printk("-t0-s=%d ms\n", cycle*1000/32768);
  146. #if 0
  147. acts_timer_set_next_irq(20);
  148. #else
  149. last_cycle_count = sys_read32(T0_CNT); /*The sleep time is not compensated */
  150. sleep_cycle_count = 0;
  151. sys_clock_set_timeout(z_get_next_timeout_expiry(), 0);
  152. #endif
  153. }
  154. } else if (action == PM_DEVICE_ACTION_SUSPEND){
  155. sys_write32(0x4, CMU_TIMER0CLK); //select rc32k
  156. sleep_st_cycle = sys_read32(T0_CNT);
  157. if(sleep_st_cycle == 0)
  158. sleep_st_cycle = 1;
  159. }
  160. return 0;
  161. }
  162. #endif
  163. uint32_t sys_clock_elapsed(void)
  164. {
  165. #if defined(CONFIG_TICKLESS_KERNEL)
  166. k_spinlock_key_t key = k_spin_lock(&lock);
  167. uint32_t cyc = acts_timer_elapsed() + cycle_count - announced_cycles;
  168. k_spin_unlock(&lock, key);
  169. return cyc / CYC_PER_TICK;
  170. #else
  171. return 0 ;
  172. #endif
  173. }
  174. uint32_t sys_clock_cycle_get_32(void)
  175. {
  176. #ifdef USE_T2_FOR_CYCLE
  177. return sys_read32(T2_CNT);
  178. #else
  179. k_spinlock_key_t key = k_spin_lock(&lock);
  180. uint32_t ret = acts_timer_elapsed() + cycle_count ;
  181. k_spin_unlock(&lock, key);
  182. return ret;
  183. #endif
  184. }
  185. void sys_clock_idle_exit(void)
  186. {
  187. }
  188. void sys_clock_disable(void)
  189. {
  190. }