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 void timer_cyc_init(void)
  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. #endif
  57. static void acts_timer_set_next_irq(uint32_t cycle)
  58. {
  59. if(cycle < CYC_PER_TICK/100) //min: 10us
  60. cycle = CYC_PER_TICK/100;
  61. if(cycle > MAX_CYCLES)
  62. cycle = MAX_CYCLES;
  63. sys_write32(sys_read32(T0_CNT)+cycle, T0_VAL);
  64. }
  65. /* Callout out of platform assembly, not hooked via IRQ_CONNECT... */
  66. void acts_timer_isr(void *arg)
  67. {
  68. ARG_UNUSED(arg);
  69. uint32_t dticks, cycle;
  70. /* Increment the amount of HW cycles elapsed (complete counter
  71. * cycles) and announce the progress to the kernel.
  72. */
  73. sys_write32(sys_read32(T0_CTL), T0_CTL); // clear penddig;
  74. cycle = sys_read32(T0_CNT);
  75. cycle_count += cycle-last_cycle_count;
  76. last_cycle_count = cycle;
  77. if(sleep_cycle_count){
  78. cycle_count += sleep_cycle_count;
  79. sleep_cycle_count = 0;
  80. }
  81. #if defined(CONFIG_TICKLESS_KERNEL)
  82. dticks = (cycle_count - announced_cycles) / CYC_PER_TICK;
  83. announced_cycles += dticks * CYC_PER_TICK;
  84. //acts_timer_set_next_irq(sys_clock_hw_cycles_per_sec()*10); //Guarantee 10 seconds of interruption
  85. sys_clock_announce(dticks);
  86. #else
  87. acts_timer_set_next_irq(CYC_PER_TICK);
  88. sys_clock_announce(1);
  89. #endif
  90. //printk("-ie=0x%x, 0x%x, 0x%x\n", cycle_count, sys_read32(T0_VAL), sys_read32(T0_CNT));
  91. }
  92. int sys_clock_driver_init(const struct device *device)
  93. {
  94. /* init timer0 as clock event device */
  95. sys_write32(0x0, CMU_TIMER0CLK); //select hosc
  96. #ifdef CONFIG_SOC_SERIES_LEOPARD
  97. acts_clock_peripheral_enable(CLOCK_ID_TIMER0);
  98. #else
  99. acts_clock_peripheral_enable(CLOCK_ID_TIMER);
  100. #endif
  101. sys_write32(0x1, T0_CTL); //clear pending stop timer
  102. timer_reg_wait();
  103. sys_write32(0xe22, T0_CTL); /* enable counter up, continue mode, irq */
  104. #ifdef USE_T2_FOR_CYCLE
  105. timer_cyc_init();
  106. #endif
  107. acts_timer_set_next_irq(CYC_PER_TICK);
  108. //printk("sys_clock_driver_init, cnt=0x%x, val=0x%x\n", sys_read32(T0_CNT), sys_read32(T0_VAL));
  109. IRQ_CONNECT(IRQ_ID_TIMER0, 0, acts_timer_isr, 0, 0);
  110. irq_enable(IRQ_ID_TIMER0);
  111. return 0;
  112. }
  113. #define RC32K_MUTIPLE soc_rc32K_mutiple_hosc()
  114. void sys_clock_set_timeout(int32_t ticks, bool idle)
  115. {
  116. #if defined(CONFIG_TICKLESS_KERNEL)
  117. k_spinlock_key_t key = k_spin_lock(&lock);
  118. uint32_t delay;
  119. ticks = (ticks == K_TICKS_FOREVER) ? MAX_TICKS : ticks;
  120. if(!ticks)
  121. ticks = 1;
  122. uint32_t unannounced = cycle_count - announced_cycles + acts_timer_elapsed();
  123. /* Desired delay in the future */
  124. delay = ticks * CYC_PER_TICK;
  125. if(unannounced > delay){
  126. acts_timer_set_next_irq(20); // now is irq
  127. }else{
  128. delay = delay - unannounced + 1;
  129. acts_timer_set_next_irq(delay);
  130. }
  131. k_spin_unlock(&lock, key);
  132. #endif
  133. }
  134. #if defined(CONFIG_PM_DEVICE)
  135. #include <wait_q.h>
  136. int sys_clock_device_ctrl(const struct device *device, enum pm_device_action action)
  137. {
  138. if(action == PM_DEVICE_ACTION_RESUME){
  139. uint32_t cycle = 0;
  140. if(sleep_st_cycle){
  141. cycle = sys_read32(T0_CNT);
  142. sys_write32(0x0, CMU_TIMER0CLK); //select hosc
  143. cycle -= sleep_st_cycle;
  144. sleep_cycle_count = cycle*RC32K_MUTIPLE;
  145. sleep_st_cycle = 0;
  146. //printk("-t0-s=%d ms\n", cycle*1000/32768);
  147. #if 0
  148. acts_timer_set_next_irq(20);
  149. #else
  150. last_cycle_count = sys_read32(T0_CNT); /*The sleep time is not compensated */
  151. sleep_cycle_count = 0;
  152. sys_clock_set_timeout(z_get_next_timeout_expiry(), 0);
  153. #endif
  154. }
  155. } else if (action == PM_DEVICE_ACTION_SUSPEND){
  156. sys_write32(0x4, CMU_TIMER0CLK); //select rc32k
  157. sleep_st_cycle = sys_read32(T0_CNT);
  158. if(sleep_st_cycle == 0)
  159. sleep_st_cycle = 1;
  160. }
  161. return 0;
  162. }
  163. #endif
  164. uint32_t sys_clock_elapsed(void)
  165. {
  166. #if defined(CONFIG_TICKLESS_KERNEL)
  167. k_spinlock_key_t key = k_spin_lock(&lock);
  168. uint32_t cyc = acts_timer_elapsed() + cycle_count - announced_cycles;
  169. k_spin_unlock(&lock, key);
  170. return cyc / CYC_PER_TICK;
  171. #else
  172. return 0 ;
  173. #endif
  174. }
  175. uint32_t sys_clock_cycle_get_32(void)
  176. {
  177. #ifdef USE_T2_FOR_CYCLE
  178. return sys_read32(T2_CNT) + soc_sleep_cycle();
  179. #else
  180. k_spinlock_key_t key = k_spin_lock(&lock);
  181. uint32_t ret = acts_timer_elapsed() + cycle_count ;
  182. k_spin_unlock(&lock, key);
  183. return ret;
  184. #endif
  185. }
  186. void sys_clock_idle_exit(void)
  187. {
  188. }
  189. void sys_clock_disable(void)
  190. {
  191. }