timer_ll.h 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423
  1. // Copyright 2015-2019 Espressif Systems (Shanghai) PTE LTD
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. // The LL layer for Timer Group register operations.
  15. // Note that most of the register operations in this layer are non-atomic operations.
  16. #pragma once
  17. #ifdef __cplusplus
  18. extern "C" {
  19. #endif
  20. #include <stdlib.h>
  21. #include "hal/timer_types.h"
  22. #include "soc/timer_periph.h"
  23. _Static_assert(TIMER_INTR_T0 == TIMG_T0_INT_CLR, "Add mapping to LL interrupt handling, since it's no longer naturally compatible with the timer_intr_t");
  24. _Static_assert(TIMER_INTR_T1 == TIMG_T1_INT_CLR, "Add mapping to LL interrupt handling, since it's no longer naturally compatible with the timer_intr_t");
  25. _Static_assert(TIMER_INTR_WDT == TIMG_WDT_INT_CLR, "Add mapping to LL interrupt handling, since it's no longer naturally compatible with the timer_intr_t");
  26. // Get timer group instance with giving group number
  27. #define TIMER_LL_GET_HW(num) ((num == 0) ? (&TIMERG0) : (&TIMERG1))
  28. /**
  29. * @brief Set timer clock prescale value
  30. *
  31. * @param hw Beginning address of the peripheral registers.
  32. * @param timer_num The timer number
  33. * @param divider Prescale value (0 is not valid)
  34. *
  35. * @return None
  36. */
  37. static inline void timer_ll_set_divider(timg_dev_t *hw, timer_idx_t timer_num, uint32_t divider)
  38. {
  39. // refer to TRM 12.2.1
  40. if (divider == 65536) {
  41. divider = 0;
  42. }
  43. int timer_en = hw->hw_timer[timer_num].config.enable;
  44. hw->hw_timer[timer_num].config.enable = 0;
  45. hw->hw_timer[timer_num].config.divider = divider;
  46. hw->hw_timer[timer_num].config.enable = timer_en;
  47. }
  48. /**
  49. * @brief Get timer clock prescale value
  50. *
  51. * @param hw Beginning address of the peripheral registers.
  52. * @param timer_num The timer number
  53. * @param divider Pointer to accept the prescale value
  54. *
  55. * @return None
  56. */
  57. static inline void timer_ll_get_divider(timg_dev_t *hw, timer_idx_t timer_num, uint32_t *divider)
  58. {
  59. uint32_t d = hw->hw_timer[timer_num].config.divider;
  60. if (d == 0) {
  61. d = 65536;
  62. }
  63. *divider = d;
  64. }
  65. /**
  66. * @brief Load counter value into time-base counter
  67. *
  68. * @param hw Beginning address of the peripheral registers.
  69. * @param timer_num The timer number
  70. * @param load_val Counter value
  71. *
  72. * @return None
  73. */
  74. static inline void timer_ll_set_counter_value(timg_dev_t *hw, timer_idx_t timer_num, uint64_t load_val)
  75. {
  76. hw->hw_timer[timer_num].load_high = (uint32_t) (load_val >> 32);
  77. hw->hw_timer[timer_num].load_low = (uint32_t) load_val;
  78. hw->hw_timer[timer_num].reload = 1;
  79. }
  80. /**
  81. * @brief Get counter value from time-base counter
  82. *
  83. * @param hw Beginning address of the peripheral registers.
  84. * @param timer_num The timer number
  85. * @param timer_val Pointer to accept the counter value
  86. *
  87. * @return None
  88. */
  89. FORCE_INLINE_ATTR void timer_ll_get_counter_value(timg_dev_t *hw, timer_idx_t timer_num, uint64_t *timer_val)
  90. {
  91. hw->hw_timer[timer_num].update.update = 1;
  92. while (hw->hw_timer[timer_num].update.update) {}
  93. *timer_val = ((uint64_t) hw->hw_timer[timer_num].cnt_high << 32) | (hw->hw_timer[timer_num].cnt_low);
  94. }
  95. /**
  96. * @brief Set counter mode, include increment mode and decrement mode.
  97. *
  98. * @param hw Beginning address of the peripheral registers.
  99. * @param timer_num The timer number
  100. * @param increase_en True to increment mode, fasle to decrement mode
  101. *
  102. * @return None
  103. */
  104. static inline void timer_ll_set_counter_increase(timg_dev_t *hw, timer_idx_t timer_num, bool increase_en)
  105. {
  106. hw->hw_timer[timer_num].config.increase = increase_en;
  107. }
  108. /**
  109. * @brief Get counter mode, include increment mode and decrement mode.
  110. *
  111. * @param hw Beginning address of the peripheral registers.
  112. * @param timer_num The timer number
  113. *
  114. * @return
  115. * - true Increment mode
  116. * - false Decrement mode
  117. */
  118. static inline bool timer_ll_get_counter_increase(timg_dev_t *hw, timer_idx_t timer_num)
  119. {
  120. return hw->hw_timer[timer_num].config.increase;
  121. }
  122. /**
  123. * @brief Set counter status, enable or disable counter.
  124. *
  125. * @param hw Beginning address of the peripheral registers.
  126. * @param timer_num The timer number
  127. * @param counter_en True to enable counter, false to disable counter
  128. *
  129. * @return None
  130. */
  131. FORCE_INLINE_ATTR void timer_ll_set_counter_enable(timg_dev_t *hw, timer_idx_t timer_num, bool counter_en)
  132. {
  133. hw->hw_timer[timer_num].config.enable = counter_en;
  134. }
  135. /**
  136. * @brief Get counter status.
  137. *
  138. * @param hw Beginning address of the peripheral registers.
  139. * @param timer_num The timer number
  140. *
  141. * @return
  142. * - true Enable counter
  143. * - false Disable conuter
  144. */
  145. static inline bool timer_ll_get_counter_enable(timg_dev_t *hw, timer_idx_t timer_num)
  146. {
  147. return hw->hw_timer[timer_num].config.enable;
  148. }
  149. /**
  150. * @brief Set auto reload mode.
  151. *
  152. * @param hw Beginning address of the peripheral registers.
  153. * @param timer_num The timer number
  154. * @param auto_reload_en True to enable auto reload mode, flase to disable auto reload mode
  155. *
  156. * @return None
  157. */
  158. static inline void timer_ll_set_auto_reload(timg_dev_t *hw, timer_idx_t timer_num, bool auto_reload_en)
  159. {
  160. hw->hw_timer[timer_num].config.autoreload = auto_reload_en;
  161. }
  162. /**
  163. * @brief Get auto reload mode.
  164. *
  165. * @param hw Beginning address of the peripheral registers.
  166. * @param timer_num The timer number
  167. *
  168. * @return
  169. * - true Enable auto reload mode
  170. * - false Disable auto reload mode
  171. */
  172. FORCE_INLINE_ATTR bool timer_ll_get_auto_reload(timg_dev_t *hw, timer_idx_t timer_num)
  173. {
  174. return hw->hw_timer[timer_num].config.autoreload;
  175. }
  176. /**
  177. * @brief Set the counter value to trigger the alarm.
  178. *
  179. * @param hw Beginning address of the peripheral registers.
  180. * @param timer_num The timer number
  181. * @param alarm_value Counter value to trigger the alarm
  182. *
  183. * @return None
  184. */
  185. FORCE_INLINE_ATTR void timer_ll_set_alarm_value(timg_dev_t *hw, timer_idx_t timer_num, uint64_t alarm_value)
  186. {
  187. hw->hw_timer[timer_num].alarm_high = (uint32_t) (alarm_value >> 32);
  188. hw->hw_timer[timer_num].alarm_low = (uint32_t) alarm_value;
  189. }
  190. /**
  191. * @brief Get the counter value to trigger the alarm.
  192. *
  193. * @param hw Beginning address of the peripheral registers.
  194. * @param timer_num The timer number
  195. * @param alarm_value Pointer to accept the counter value to trigger the alarm
  196. *
  197. * @return None
  198. */
  199. static inline void timer_ll_get_alarm_value(timg_dev_t *hw, timer_idx_t timer_num, uint64_t *alarm_value)
  200. {
  201. *alarm_value = ((uint64_t) hw->hw_timer[timer_num].alarm_high << 32) | (hw->hw_timer[timer_num].alarm_low);
  202. }
  203. /**
  204. * @brief Set the alarm status, enable or disable the alarm.
  205. *
  206. * @param hw Beginning address of the peripheral registers.
  207. * @param timer_num The timer number
  208. * @param alarm_en True to enable alarm, false to disable alarm
  209. *
  210. * @return None
  211. */
  212. FORCE_INLINE_ATTR void timer_ll_set_alarm_enable(timg_dev_t *hw, timer_idx_t timer_num, bool alarm_en)
  213. {
  214. hw->hw_timer[timer_num].config.alarm_en = alarm_en;
  215. }
  216. /**
  217. * @brief Get the alarm status.
  218. *
  219. * @param hw Beginning address of the peripheral registers.
  220. * @param timer_num The timer number
  221. *
  222. * @return
  223. * - true Enable alarm
  224. * - false Disable alarm
  225. */
  226. static inline bool timer_ll_get_alarm_enable(timg_dev_t *hw, timer_idx_t timer_num)
  227. {
  228. return hw->hw_timer[timer_num].config.alarm_en;
  229. }
  230. /**
  231. * @brief Enable timer interrupt.
  232. *
  233. * @param hw Beginning address of the peripheral registers.
  234. * @param timer_num The timer number
  235. *
  236. * @return None
  237. */
  238. FORCE_INLINE_ATTR void timer_ll_intr_enable(timg_dev_t *hw, timer_idx_t timer_num)
  239. {
  240. hw->int_ena.val |= BIT(timer_num);
  241. hw->hw_timer[timer_num].config.level_int_en = 1;
  242. }
  243. /**
  244. * @brief Disable timer interrupt.
  245. *
  246. * @param hw Beginning address of the peripheral registers.
  247. * @param timer_num The timer number
  248. *
  249. * @return None
  250. */
  251. FORCE_INLINE_ATTR void timer_ll_intr_disable(timg_dev_t *hw, timer_idx_t timer_num)
  252. {
  253. hw->int_ena.val &= (~BIT(timer_num));
  254. hw->hw_timer[timer_num].config.level_int_en = 0;
  255. }
  256. /**
  257. * @brief Disable timer interrupt.
  258. *
  259. * @param hw Beginning address of the peripheral registers.
  260. * @param timer_num The timer number
  261. *
  262. * @return None
  263. */
  264. FORCE_INLINE_ATTR void timer_ll_clear_intr_status(timg_dev_t *hw, timer_idx_t timer_num)
  265. {
  266. hw->int_clr.val |= BIT(timer_num);
  267. }
  268. /**
  269. * @brief Get interrupt status.
  270. *
  271. * @param hw Beginning address of the peripheral registers.
  272. * @param intr_status Interrupt status
  273. *
  274. * @return None
  275. */
  276. FORCE_INLINE_ATTR void timer_ll_get_intr_status(timg_dev_t *hw, uint32_t *intr_status)
  277. {
  278. *intr_status = hw->int_st.val & 0x03;
  279. }
  280. /**
  281. * @brief Get interrupt raw status.
  282. *
  283. * @param group_num Timer group number, 0 for TIMERG0 or 1 for TIMERG1
  284. * @param intr_raw_status Interrupt raw status
  285. *
  286. * @return None
  287. */
  288. FORCE_INLINE_ATTR void timer_ll_get_intr_raw_status(timer_group_t group_num, uint32_t *intr_raw_status)
  289. {
  290. timg_dev_t *hw = TIMER_LL_GET_HW(group_num);
  291. *intr_raw_status = hw->int_raw.val & 0x03;
  292. }
  293. /**
  294. * @brief Set the level interrupt status, enable or disable the level interrupt.
  295. *
  296. * @param hw Beginning address of the peripheral registers.
  297. * @param timer_num The timer number
  298. * @param level_int_en True to enable level interrupt, false to disable level interrupt
  299. *
  300. * @return None
  301. */
  302. static inline void timer_ll_set_level_int_enable(timg_dev_t *hw, timer_idx_t timer_num, bool level_int_en)
  303. {
  304. hw->hw_timer[timer_num].config.level_int_en = level_int_en;
  305. }
  306. /**
  307. * @brief Get the level interrupt status.
  308. *
  309. * @param hw Beginning address of the peripheral registers.
  310. * @param timer_num The timer number
  311. *
  312. * @return
  313. * - true Enable level interrupt
  314. * - false Disable level interrupt
  315. */
  316. static inline bool timer_ll_get_level_int_enable(timg_dev_t *hw, timer_idx_t timer_num)
  317. {
  318. return hw->hw_timer[timer_num].config.level_int_en;
  319. }
  320. /**
  321. * @brief Set the edge interrupt status, enable or disable the edge interrupt.
  322. *
  323. * @param hw Beginning address of the peripheral registers.
  324. * @param timer_num The timer number
  325. * @param edge_int_en True to enable edge interrupt, false to disable edge interrupt
  326. *
  327. * @return None
  328. */
  329. static inline void timer_ll_set_edge_int_enable(timg_dev_t *hw, timer_idx_t timer_num, bool edge_int_en)
  330. {
  331. hw->hw_timer[timer_num].config.edge_int_en = edge_int_en;
  332. }
  333. /**
  334. * @brief Get the edge interrupt status.
  335. *
  336. * @param hw Beginning address of the peripheral registers.
  337. * @param timer_num The timer number
  338. *
  339. * @return
  340. * - true Enable edge interrupt
  341. * - false Disable edge interrupt
  342. */
  343. static inline bool timer_ll_get_edge_int_enable(timg_dev_t *hw, timer_idx_t timer_num)
  344. {
  345. return hw->hw_timer[timer_num].config.edge_int_en;
  346. }
  347. /**
  348. * @brief Get interrupt status register address.
  349. *
  350. * @param hw Beginning address of the peripheral registers.
  351. *
  352. * @return uint32_t Interrupt status register address
  353. */
  354. static inline uint32_t timer_ll_get_intr_status_reg(timg_dev_t *hw)
  355. {
  356. return (uint32_t) & (hw->int_st.val);
  357. }
  358. static inline uint32_t timer_ll_get_intr_mask_bit(timg_dev_t *hw, timer_idx_t timer_num)
  359. {
  360. return (1U << timer_num);
  361. }
  362. /**
  363. * @brief Set clock source.
  364. *
  365. * @param hal Context of the HAL layer
  366. * @param use_xtal_en True to use XTAL clock, flase to use APB clock
  367. *
  368. * @return None
  369. */
  370. static inline void timer_ll_set_use_xtal(timg_dev_t *hw, timer_idx_t timer_num, bool use_xtal_en)
  371. {
  372. hw->hw_timer[timer_num].config.use_xtal = use_xtal_en;
  373. }
  374. /**
  375. * @brief Get clock source.
  376. *
  377. * @param hal Context of the HAL layer
  378. *
  379. * @return
  380. * - true Use XTAL clock
  381. * - false Use APB clock
  382. */
  383. static inline bool timer_ll_get_use_xtal(timg_dev_t *hw, timer_idx_t timer_num)
  384. {
  385. return hw->hw_timer[timer_num].config.use_xtal;
  386. }
  387. #ifdef __cplusplus
  388. }
  389. #endif