drv_rtc.c 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278
  1. /*
  2. * Copyright (c) 2006-2025, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2025-08-19 Alex Yang Add MCXA346 RTC driver for RT-Thread
  9. */
  10. #include <rtthread.h>
  11. #include <rtdevice.h>
  12. #include <sys/time.h>
  13. #ifdef BSP_USING_RTC
  14. #define DBG_TAG "drv.rtc"
  15. #define DBG_LVL DBG_INFO
  16. #include <rtdbg.h>
  17. #include "fsl_rtc.h"
  18. #include "fsl_clock.h"
  19. /* Get RTC timestamp */
  20. static time_t get_rtc_timestamp(void)
  21. {
  22. rtc_datetime_t datetime;
  23. struct tm tm_new;
  24. /* Get current time from RTC */
  25. RTC_GetDatetime(RTC0, &datetime);
  26. tm_new.tm_sec = datetime.second;
  27. tm_new.tm_min = datetime.minute;
  28. tm_new.tm_hour = datetime.hour;
  29. tm_new.tm_mday = datetime.day;
  30. tm_new.tm_mon = datetime.month - 1;
  31. tm_new.tm_year = datetime.year - 1900;
  32. tm_new.tm_isdst = 0;
  33. LOG_D("get rtc time: %04d-%02d-%02d %02d:%02d:%02d",
  34. datetime.year, datetime.month, datetime.day,
  35. datetime.hour, datetime.minute, datetime.second);
  36. return mktime(&tm_new);
  37. }
  38. /* Set RTC timestamp */
  39. static rt_err_t set_rtc_time_stamp(time_t time_stamp)
  40. {
  41. rtc_datetime_t datetime;
  42. struct tm *time_tm;
  43. time_tm = gmtime(&time_stamp);
  44. if (time_tm->tm_year < 70) /* Year should be >= 1970 */
  45. {
  46. LOG_E("Invalid year: %d", time_tm->tm_year + 1900);
  47. return -RT_ERROR;
  48. }
  49. /* Convert to RTC datetime format */
  50. datetime.year = time_tm->tm_year + 1900;
  51. datetime.month = time_tm->tm_mon + 1;
  52. datetime.day = time_tm->tm_mday;
  53. datetime.hour = time_tm->tm_hour;
  54. datetime.minute = time_tm->tm_min;
  55. datetime.second = time_tm->tm_sec;
  56. /* Set RTC time */
  57. RTC_StopTimer(RTC0);
  58. RTC_SetDatetime(RTC0, &datetime);
  59. RTC_StartTimer(RTC0);
  60. LOG_D("set rtc time: %04d-%02d-%02d %02d:%02d:%02d",
  61. datetime.year, datetime.month, datetime.day,
  62. datetime.hour, datetime.minute, datetime.second);
  63. return RT_EOK;
  64. }
  65. /* RTC configuration */
  66. static rt_err_t rt_rtc_config(void)
  67. {
  68. rtc_config_t rtc_config;
  69. /* Get default RTC configuration */
  70. RTC_GetDefaultConfig(&rtc_config);
  71. /* Initialize RTC - Note: RTC_Init returns void, not status */
  72. RTC_Init(RTC0, &rtc_config);
  73. /* Start RTC timer */
  74. RTC_StartTimer(RTC0);
  75. return RT_EOK;
  76. }
  77. /* RTC initialization */
  78. static rt_err_t _rtc_init(void)
  79. {
  80. /* Configure RTC */
  81. if (rt_rtc_config() != RT_EOK)
  82. {
  83. LOG_E("RTC config failed.");
  84. return -RT_ERROR;
  85. }
  86. LOG_D("RTC initialized successfully");
  87. return RT_EOK;
  88. }
  89. /* Get RTC seconds */
  90. static rt_err_t _rtc_get_secs(time_t *args)
  91. {
  92. RT_ASSERT(args != RT_NULL);
  93. *args = get_rtc_timestamp();
  94. LOG_D("RTC: get rtc_time %x", *args);
  95. return RT_EOK;
  96. }
  97. /* Set RTC seconds */
  98. static rt_err_t _rtc_set_secs(time_t *args)
  99. {
  100. rt_err_t result = RT_EOK;
  101. RT_ASSERT(args != RT_NULL);
  102. if (set_rtc_time_stamp(*args) != RT_EOK)
  103. {
  104. result = -RT_ERROR;
  105. }
  106. LOG_D("RTC: set rtc_time %x", *args);
  107. return result;
  108. }
  109. /* Get RTC alarm */
  110. static rt_err_t _rtc_get_alarm(struct rt_rtc_wkalarm *wkalarm)
  111. {
  112. rtc_datetime_t datetime;
  113. RT_ASSERT(wkalarm != RT_NULL);
  114. /* Get alarm time from RTC */
  115. RTC_GetAlarm(RTC0, &datetime);
  116. /* Convert to wkalarm format */
  117. wkalarm->tm_sec = datetime.second;
  118. wkalarm->tm_min = datetime.minute;
  119. wkalarm->tm_hour = datetime.hour;
  120. wkalarm->tm_mday = datetime.day;
  121. wkalarm->tm_mon = datetime.month - 1;
  122. wkalarm->tm_year = datetime.year - 1900;
  123. /* Check if alarm is enabled */
  124. wkalarm->enable = (RTC_GetEnabledInterrupts(RTC0) & kRTC_AlarmInterruptEnable) ? 1 : 0;
  125. LOG_D("RTC: get alarm %04d-%02d-%02d %02d:%02d:%02d (%s)",
  126. datetime.year, datetime.month, datetime.day,
  127. datetime.hour, datetime.minute, datetime.second,
  128. wkalarm->enable ? "ENABLED" : "DISABLED");
  129. return RT_EOK;
  130. }
  131. /* Set RTC alarm */
  132. static rt_err_t _rtc_set_alarm(struct rt_rtc_wkalarm *wkalarm)
  133. {
  134. rtc_datetime_t datetime;
  135. RT_ASSERT(wkalarm != RT_NULL);
  136. /* Convert from wkalarm format */
  137. datetime.year = wkalarm->tm_year + 1900;
  138. datetime.month = wkalarm->tm_mon + 1;
  139. datetime.day = wkalarm->tm_mday;
  140. datetime.hour = wkalarm->tm_hour;
  141. datetime.minute = wkalarm->tm_min;
  142. datetime.second = wkalarm->tm_sec;
  143. /* Set alarm time */
  144. RTC_SetAlarm(RTC0, &datetime);
  145. /* Enable/disable alarm interrupt */
  146. if (wkalarm->enable)
  147. {
  148. RTC_EnableInterrupts(RTC0, kRTC_AlarmInterruptEnable);
  149. EnableIRQ(RTC_IRQn); /* Use RTC_IRQn instead of RTC0_IRQn */
  150. LOG_D("RTC alarm enabled");
  151. }
  152. else
  153. {
  154. RTC_DisableInterrupts(RTC0, kRTC_AlarmInterruptEnable);
  155. LOG_D("RTC alarm disabled");
  156. }
  157. LOG_D("RTC: set alarm %04d-%02d-%02d %02d:%02d:%02d",
  158. datetime.year, datetime.month, datetime.day,
  159. datetime.hour, datetime.minute, datetime.second);
  160. return RT_EOK;
  161. }
  162. /* RTC operations structure */
  163. static const struct rt_rtc_ops _rtc_ops =
  164. {
  165. _rtc_init,
  166. _rtc_get_secs,
  167. _rtc_set_secs,
  168. _rtc_get_alarm,
  169. _rtc_set_alarm,
  170. RT_NULL, /* get_timeval */
  171. RT_NULL, /* set_timeval */
  172. };
  173. static rt_rtc_dev_t mcxa_rtc_dev;
  174. /* RTC interrupt handler */
  175. void RTC_IRQHandler(void)
  176. {
  177. rt_interrupt_enter();
  178. /* Get interrupt status */
  179. uint32_t status = RTC_GetStatusFlags(RTC0);
  180. /* Handle alarm interrupt */
  181. if (status & kRTC_AlarmFlag)
  182. {
  183. /* Clear alarm flag */
  184. RTC_ClearStatusFlags(RTC0, kRTC_AlarmFlag);
  185. LOG_D("RTC alarm triggered");
  186. /* If alarm framework is available, notify it */
  187. #ifdef RT_USING_ALARM
  188. /* Send alarm event to alarm thread */
  189. rt_event_send(&_container.event, 1);
  190. #endif
  191. }
  192. /* Handle seconds interrupt if needed */
  193. if (status & kRTC_SecondsInterruptEnable)
  194. {
  195. LOG_D("RTC seconds interrupt");
  196. }
  197. rt_interrupt_leave();
  198. }
  199. /* Hardware RTC initialization */
  200. int rt_hw_rtc_init(void)
  201. {
  202. rt_err_t result;
  203. /* Set RTC operations */
  204. mcxa_rtc_dev.ops = &_rtc_ops;
  205. /* Register RTC device */
  206. result = rt_hw_rtc_register(&mcxa_rtc_dev, "rtc", RT_DEVICE_FLAG_RDWR, RT_NULL);
  207. if (result != RT_EOK)
  208. {
  209. LOG_E("RTC register failed, err code: %d", result);
  210. return result;
  211. }
  212. LOG_D("RTC init success");
  213. return RT_EOK;
  214. }
  215. INIT_DEVICE_EXPORT(rt_hw_rtc_init);
  216. #endif /* BSP_USING_RTC */