gd32f3x0_rtc.c 35 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966
  1. /*!
  2. \file gd32f3x0_rtc.c
  3. \brief RTC driver
  4. \version 2017-06-06, V1.0.0, firmware for GD32F3x0
  5. \version 2019-06-01, V2.0.0, firmware for GD32F3x0
  6. */
  7. /*
  8. Copyright (c) 2019, GigaDevice Semiconductor Inc.
  9. Redistribution and use in source and binary forms, with or without modification,
  10. are permitted provided that the following conditions are met:
  11. 1. Redistributions of source code must retain the above copyright notice, this
  12. list of conditions and the following disclaimer.
  13. 2. Redistributions in binary form must reproduce the above copyright notice,
  14. this list of conditions and the following disclaimer in the documentation
  15. and/or other materials provided with the distribution.
  16. 3. Neither the name of the copyright holder nor the names of its contributors
  17. may be used to endorse or promote products derived from this software without
  18. specific prior written permission.
  19. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  20. AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
  21. WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
  22. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
  23. INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
  24. NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
  25. PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
  26. WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
  27. ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY
  28. OF SUCH DAMAGE.
  29. */
  30. #include "gd32f3x0_rtc.h"
  31. /*!
  32. \brief reset most of the RTC registers
  33. \param[in] none
  34. \param[out] none
  35. \retval ErrStatus: ERROR or SUCCESS
  36. */
  37. ErrStatus rtc_deinit(void)
  38. {
  39. ErrStatus error_status = ERROR;
  40. /* RTC_TAMP register is not under write protection */
  41. RTC_TAMP = RTC_REGISTER_RESET;
  42. /* disable the write protection */
  43. RTC_WPK = RTC_UNLOCK_KEY1;
  44. RTC_WPK = RTC_UNLOCK_KEY2;
  45. /* reset RTC_CTL register, this can be done without the init mode */
  46. RTC_CTL &= RTC_REGISTER_RESET;
  47. /* enter init mode */
  48. error_status = rtc_init_mode_enter();
  49. if(ERROR != error_status){
  50. /* before reset RTC_TIME and RTC_DATE, BPSHAD bit in RTC_CTL should be reset as the condition.
  51. in order to read calendar from shadow register, not the real registers being reset */
  52. RTC_TIME = RTC_REGISTER_RESET;
  53. RTC_DATE = RTC_DATE_RESET;
  54. RTC_PSC = RTC_PSC_RESET;
  55. /* reset RTC_STAT register, also exit init mode.
  56. at the same time, RTC_STAT_SOPF bit is reset, as the condition to reset RTC_SHIFTCTL register later */
  57. RTC_STAT = RTC_STAT_RESET;
  58. /* to write RTC_ALRM0SS register, ALRM0EN bit in RTC_CTL register should be reset as the condition */
  59. RTC_ALRM0TD = RTC_REGISTER_RESET;
  60. RTC_ALRM0SS = RTC_REGISTER_RESET;
  61. /* reset RTC_SHIFTCTL and RTC_HRFC register, this can be done without the init mode */
  62. RTC_SHIFTCTL = RTC_REGISTER_RESET;
  63. RTC_HRFC = RTC_REGISTER_RESET;
  64. error_status = rtc_register_sync_wait();
  65. }
  66. /* enable the write protection */
  67. RTC_WPK = RTC_LOCK_KEY;
  68. return error_status;
  69. }
  70. /*!
  71. \brief initialize RTC registers
  72. \param[in] rtc_initpara_struct: pointer to a rtc_parameter_struct structure which contains
  73. parameters for initialization of the rtc peripheral
  74. members of the structure and the member values are shown as below:
  75. rtc_year: 0x0 - 0x99(BCD format)
  76. rtc_month: RTC_JAN, RTC_FEB, RTC_MAR, RTC_APR, RTC_MAY, RTC_JUN,
  77. RTC_JUL, RTC_AUG, RTC_SEP, RTC_OCT, RTC_NOV, RTC_DEC
  78. rtc_date: 0x1 - 0x31(BCD format)
  79. rtc_day_of_week: RTC_MONDAY, RTC_TUESDAY, RTC_WEDSDAY, RTC_THURSDAY
  80. RTC_FRIDAY, RTC_SATURDAY, RTC_SUNDAY
  81. rtc_hour: 0x0 - 0x12(BCD format) or 0x0 - 0x23(BCD format) depending on the rtc_display_format chose
  82. rtc_minute: 0x0 - 0x59(BCD format)
  83. rtc_second: 0x0 - 0x59(BCD format)
  84. rtc_factor_asyn: 0x0 - 0x7F
  85. rtc_factor_syn: 0x0 - 0x7FFF
  86. rtc_am_pm: RTC_AM, RTC_PM
  87. rtc_display_format: RTC_24HOUR, RTC_12HOUR
  88. \param[out] none
  89. \retval ErrStatus: ERROR or SUCCESS
  90. */
  91. ErrStatus rtc_init(rtc_parameter_struct* rtc_initpara_struct)
  92. {
  93. ErrStatus error_status = ERROR;
  94. uint32_t reg_time = 0x00U, reg_date = 0x00U;
  95. reg_date = (DATE_YR(rtc_initpara_struct->rtc_year) | \
  96. DATE_DOW(rtc_initpara_struct->rtc_day_of_week) | \
  97. DATE_MON(rtc_initpara_struct->rtc_month) | \
  98. DATE_DAY(rtc_initpara_struct->rtc_date));
  99. reg_time = (rtc_initpara_struct->rtc_am_pm| \
  100. TIME_HR(rtc_initpara_struct->rtc_hour) | \
  101. TIME_MN(rtc_initpara_struct->rtc_minute) | \
  102. TIME_SC(rtc_initpara_struct->rtc_second));
  103. /* 1st: disable the write protection */
  104. RTC_WPK = RTC_UNLOCK_KEY1;
  105. RTC_WPK = RTC_UNLOCK_KEY2;
  106. /* 2nd: enter init mode */
  107. error_status = rtc_init_mode_enter();
  108. if(ERROR != error_status){
  109. RTC_PSC = (uint32_t)(PSC_FACTOR_A(rtc_initpara_struct->rtc_factor_asyn)| \
  110. PSC_FACTOR_S(rtc_initpara_struct->rtc_factor_syn));
  111. RTC_TIME = (uint32_t)reg_time;
  112. RTC_DATE = (uint32_t)reg_date;
  113. RTC_CTL &= (uint32_t)(~RTC_CTL_CS);
  114. RTC_CTL |= rtc_initpara_struct->rtc_display_format;
  115. /* 3rd: exit init mode */
  116. rtc_init_mode_exit();
  117. /* 4th: wait the RSYNF flag to set */
  118. error_status = rtc_register_sync_wait();
  119. }
  120. /* 5th: enable the write protection */
  121. RTC_WPK = RTC_LOCK_KEY;
  122. return error_status;
  123. }
  124. /*!
  125. \brief enter RTC init mode
  126. \param[in] none
  127. \param[out] none
  128. \retval ErrStatus: ERROR or SUCCESS
  129. */
  130. ErrStatus rtc_init_mode_enter(void)
  131. {
  132. uint32_t time_index = RTC_INITM_TIMEOUT;
  133. uint32_t flag_status = RESET;
  134. ErrStatus error_status = ERROR;
  135. /* check whether it has been in init mode */
  136. if(RESET == (RTC_STAT & RTC_STAT_INITF)){
  137. RTC_STAT |= RTC_STAT_INITM;
  138. /* wait until the INITF flag to be set */
  139. do{
  140. flag_status = RTC_STAT & RTC_STAT_INITF;
  141. }while((--time_index > 0x00U) && (RESET == flag_status));
  142. if(RESET != flag_status){
  143. error_status = SUCCESS;
  144. }
  145. }else{
  146. error_status = SUCCESS;
  147. }
  148. return error_status;
  149. }
  150. /*!
  151. \brief exit RTC init mode
  152. \param[in] none
  153. \param[out] none
  154. \retval none
  155. */
  156. void rtc_init_mode_exit(void)
  157. {
  158. RTC_STAT &= (uint32_t)(~RTC_STAT_INITM);
  159. }
  160. /*!
  161. \brief wait until RTC_TIME and RTC_DATE registers are synchronized with APB clock, and the shadow
  162. registers are updated
  163. \param[in] none
  164. \param[out] none
  165. \retval ErrStatus: ERROR or SUCCESS
  166. */
  167. ErrStatus rtc_register_sync_wait(void)
  168. {
  169. volatile uint32_t time_index = RTC_RSYNF_TIMEOUT;
  170. uint32_t flag_status = RESET;
  171. ErrStatus error_status = ERROR;
  172. if(RESET == (RTC_CTL & RTC_CTL_BPSHAD)){
  173. /* disable the write protection */
  174. RTC_WPK = RTC_UNLOCK_KEY1;
  175. RTC_WPK = RTC_UNLOCK_KEY2;
  176. /* firstly clear RSYNF flag */
  177. RTC_STAT &= (uint32_t)(~RTC_STAT_RSYNF);
  178. /* wait until RSYNF flag to be set */
  179. do{
  180. flag_status = RTC_STAT & RTC_STAT_RSYNF;
  181. }while((--time_index > 0x00U) && (RESET == flag_status));
  182. if(RESET != flag_status){
  183. error_status = SUCCESS;
  184. }
  185. /* enable the write protection */
  186. RTC_WPK = RTC_LOCK_KEY;
  187. }else{
  188. error_status = SUCCESS;
  189. }
  190. return error_status;
  191. }
  192. /*!
  193. \brief get current time and date
  194. \param[in] none
  195. \param[out] rtc_initpara_struct: pointer to a rtc_parameter_struct structure which contains
  196. parameters for initialization of the rtc peripheral
  197. members of the structure and the member values are shown as below:
  198. rtc_year: 0x0 - 0x99(BCD format)
  199. rtc_month: RTC_JAN, RTC_FEB, RTC_MAR, RTC_APR, RTC_MAY, RTC_JUN,
  200. RTC_JUL, RTC_AUG, RTC_SEP, RTC_OCT, RTC_NOV, RTC_DEC
  201. rtc_date: 0x1 - 0x31(BCD format)
  202. rtc_day_of_week: RTC_MONDAY, RTC_TUESDAY, RTC_WEDSDAY, RTC_THURSDAY
  203. RTC_FRIDAY, RTC_SATURDAY, RTC_SUNDAY
  204. rtc_hour: 0x0 - 0x12(BCD format) or 0x0 - 0x23(BCD format) depending on the rtc_display_format chose
  205. rtc_minute: 0x0 - 0x59(BCD format)
  206. rtc_second: 0x0 - 0x59(BCD format)
  207. rtc_factor_asyn: 0x0 - 0x7F
  208. rtc_factor_syn: 0x0 - 0x7FFF
  209. rtc_am_pm: RTC_AM, RTC_PM
  210. rtc_display_format: RTC_24HOUR, RTC_12HOUR
  211. \retval none
  212. */
  213. void rtc_current_time_get(rtc_parameter_struct* rtc_initpara_struct)
  214. {
  215. uint32_t temp_tr = 0x00U, temp_dr = 0x00U, temp_pscr = 0x00U, temp_ctlr = 0x00U;
  216. temp_tr = (uint32_t)RTC_TIME;
  217. temp_dr = (uint32_t)RTC_DATE;
  218. temp_pscr = (uint32_t)RTC_PSC;
  219. temp_ctlr = (uint32_t)RTC_CTL;
  220. /* get current time and construct rtc_parameter_struct structure */
  221. rtc_initpara_struct->rtc_year = (uint8_t)GET_DATE_YR(temp_dr);
  222. rtc_initpara_struct->rtc_month = (uint8_t)GET_DATE_MON(temp_dr);
  223. rtc_initpara_struct->rtc_date = (uint8_t)GET_DATE_DAY(temp_dr);
  224. rtc_initpara_struct->rtc_day_of_week = (uint8_t)GET_DATE_DOW(temp_dr);
  225. rtc_initpara_struct->rtc_hour = (uint8_t)GET_TIME_HR(temp_tr);
  226. rtc_initpara_struct->rtc_minute = (uint8_t)GET_TIME_MN(temp_tr);
  227. rtc_initpara_struct->rtc_second = (uint8_t)GET_TIME_SC(temp_tr);
  228. rtc_initpara_struct->rtc_factor_asyn = (uint16_t)GET_PSC_FACTOR_A(temp_pscr);
  229. rtc_initpara_struct->rtc_factor_syn = (uint16_t)GET_PSC_FACTOR_S(temp_pscr);
  230. rtc_initpara_struct->rtc_am_pm = (uint32_t)(temp_pscr & RTC_TIME_PM);
  231. rtc_initpara_struct->rtc_display_format = (uint32_t)(temp_ctlr & RTC_CTL_CS);
  232. }
  233. /*!
  234. \brief get current subsecond value
  235. \param[in] none
  236. \param[out] none
  237. \retval current subsecond value
  238. */
  239. uint32_t rtc_subsecond_get(void)
  240. {
  241. uint32_t reg = 0x00U;
  242. /* if BPSHAD bit is reset, reading RTC_SS will lock RTC_TIME and RTC_DATE automatically */
  243. reg = (uint32_t)RTC_SS;
  244. /* read RTC_DATE to unlock the 3 shadow registers */
  245. (void) (RTC_DATE);
  246. return reg;
  247. }
  248. /*!
  249. \brief configure RTC alarm
  250. \param[in] rtc_alarm_time: pointer to a rtc_alarm_struct structure which contains
  251. parameters for RTC alarm configuration
  252. members of the structure and the member values are shown as below:
  253. rtc_alarm_mask: RTC_ALARM_NONE_MASK, RTC_ALARM_DATE_MASK, RTC_ALARM_HOUR_MASK
  254. RTC_ALARM_MINUTE_MASK, RTC_ALARM_SECOND_MASK, RTC_ALARM_ALL_MASK
  255. rtc_weekday_or_date: RTC_ALARM_DATE_SELECTED, RTC_ALARM_WEEKDAY_SELECTED
  256. rtc_alarm_day: 1) 0x1 - 0x31(BCD format) if RTC_ALARM_DATE_SELECTED is set
  257. 2) RTC_MONDAY, RTC_TUESDAY, RTC_WEDSDAY, RTC_THURSDAY, RTC_FRIDAY,
  258. RTC_SATURDAY, RTC_SUNDAY if RTC_ALARM_WEEKDAY_SELECTED is set
  259. rtc_alarm_hour: 0x0 - 0x12(BCD format) or 0x0 - 0x23(BCD format) depending on the rtc_display_format
  260. rtc_alarm_minute: 0x0 - 0x59(BCD format)
  261. rtc_alarm_second: 0x0 - 0x59(BCD format)
  262. rtc_am_pm: RTC_AM, RTC_PM
  263. \param[out] none
  264. \retval none
  265. */
  266. void rtc_alarm_config(rtc_alarm_struct* rtc_alarm_time)
  267. {
  268. uint32_t reg_alrm0td = 0x00U;
  269. reg_alrm0td = (rtc_alarm_time->rtc_alarm_mask | \
  270. rtc_alarm_time->rtc_weekday_or_date | \
  271. rtc_alarm_time->rtc_am_pm | \
  272. ALRM0TD_DAY(rtc_alarm_time->rtc_alarm_day) | \
  273. ALRM0TD_HR(rtc_alarm_time->rtc_alarm_hour) | \
  274. ALRM0TD_MN(rtc_alarm_time->rtc_alarm_minute) | \
  275. ALRM0TD_SC(rtc_alarm_time->rtc_alarm_second));
  276. /* disable the write protection */
  277. RTC_WPK = RTC_UNLOCK_KEY1;
  278. RTC_WPK = RTC_UNLOCK_KEY2;
  279. RTC_ALRM0TD = (uint32_t)reg_alrm0td;
  280. /* enable the write protection */
  281. RTC_WPK = RTC_LOCK_KEY;
  282. }
  283. /*!
  284. \brief configure subsecond of RTC alarm
  285. \param[in] mask_subsecond: alarm subsecond mask
  286. only one parameter can be selected which is shown as below:
  287. \arg RTC_MASKSSC_0_14: mask alarm subsecond configuration
  288. \arg RTC_MASKSSC_1_14: mask RTC_ALRM0SS_SSC[14:1], and RTC_ALRM0SS_SSC[0] is to be compared
  289. \arg RTC_MASKSSC_2_14: mask RTC_ALRM0SS_SSC[14:2], and RTC_ALRM0SS_SSC[1:0] is to be compared
  290. \arg RTC_MASKSSC_3_14: mask RTC_ALRM0SS_SSC[14:3], and RTC_ALRM0SS_SSC[2:0] is to be compared
  291. \arg RTC_MASKSSC_4_14: mask RTC_ALRM0SS_SSC[14:4], and RTC_ALRM0SS_SSC[3:0] is to be compared
  292. \arg RTC_MASKSSC_5_14: mask RTC_ALRM0SS_SSC[14:5], and RTC_ALRM0SS_SSC[4:0] is to be compared
  293. \arg RTC_MASKSSC_6_14: mask RTC_ALRM0SS_SSC[14:6], and RTC_ALRM0SS_SSC[5:0] is to be compared
  294. \arg RTC_MASKSSC_7_14: mask RTC_ALRM0SS_SSC[14:7], and RTC_ALRM0SS_SSC[6:0] is to be compared
  295. \arg RTC_MASKSSC_8_14: mask RTC_ALRM0SS_SSC[14:8], and RTC_ALRM0SS_SSC[7:0] is to be compared
  296. \arg RTC_MASKSSC_9_14: mask RTC_ALRM0SS_SSC[14:9], and RTC_ALRM0SS_SSC[8:0] is to be compared
  297. \arg RTC_MASKSSC_10_14: mask RTC_ALRM0SS_SSC[14:10], and RTC_ALRM0SS_SSC[9:0] is to be compared
  298. \arg RTC_MASKSSC_11_14: mask RTC_ALRM0SS_SSC[14:11], and RTC_ALRM0SS_SSC[10:0] is to be compared
  299. \arg RTC_MASKSSC_12_14: mask RTC_ALRM0SS_SSC[14:12], and RTC_ALRM0SS_SSC[11:0] is to be compared
  300. \arg RTC_MASKSSC_13_14: mask RTC_ALRM0SS_SSC[14:13], and RTC_ALRM0SS_SSC[12:0] is to be compared
  301. \arg RTC_MASKSSC_14: mask RTC_ALRM0SS_SSC[14], and RTC_ALRM0SS_SSC[13:0] is to be compared
  302. \arg RTC_MASKSSC_NONE: mask none, and RTC_ALRM0SS_SSC[14:0] is to be compared
  303. \param[in] subsecond: alarm subsecond value(0x000 - 0x7FFF)
  304. \param[out] none
  305. \retval none
  306. */
  307. void rtc_alarm_subsecond_config(uint32_t mask_subsecond, uint32_t subsecond)
  308. {
  309. /* disable the write protection */
  310. RTC_WPK = RTC_UNLOCK_KEY1;
  311. RTC_WPK = RTC_UNLOCK_KEY2;
  312. RTC_ALRM0SS = mask_subsecond | subsecond;
  313. /* enable the write protection */
  314. RTC_WPK = RTC_LOCK_KEY;
  315. }
  316. /*!
  317. \brief get RTC alarm
  318. \param[in] none
  319. \param[out] rtc_alarm_time: pointer to a rtc_alarm_struct structure which contains
  320. parameters for RTC alarm configuration
  321. members of the structure and the member values are shown as below:
  322. rtc_alarm_mask: RTC_ALARM_NONE_MASK, RTC_ALARM_DATE_MASK, RTC_ALARM_HOUR_MASK
  323. RTC_ALARM_MINUTE_MASK, RTC_ALARM_SECOND_MASK, RTC_ALARM_ALL_MASK
  324. rtc_weekday_or_date: RTC_ALARM_DATE_SELECTED, RTC_ALARM_WEEKDAY_SELECTED
  325. rtc_alarm_day: 1) 0x1 - 0x31(BCD format) if RTC_ALARM_DATE_SELECTED is set
  326. 2) RTC_MONDAY, RTC_TUESDAY, RTC_WEDSDAY, RTC_THURSDAY, RTC_FRIDAY,
  327. RTC_SATURDAY, RTC_SUNDAY if RTC_ALARM_WEEKDAY_SELECTED is set
  328. rtc_alarm_hour: 0x0 - 0x12(BCD format) or 0x0 - 0x23(BCD format) depending on the rtc_display_format
  329. rtc_alarm_minute: 0x0 - 0x59(BCD format)
  330. rtc_alarm_second: 0x0 - 0x59(BCD format)
  331. rtc_am_pm: RTC_AM, RTC_PM
  332. \retval none
  333. */
  334. void rtc_alarm_get(rtc_alarm_struct* rtc_alarm_time)
  335. {
  336. uint32_t reg_alrm0td = 0x00U;
  337. /* get the value of RTC_ALRM0TD register */
  338. reg_alrm0td = RTC_ALRM0TD;
  339. /* get alarm parameters and construct the rtc_alarm_struct structure */
  340. rtc_alarm_time->rtc_alarm_mask = reg_alrm0td & RTC_ALARM_ALL_MASK;
  341. rtc_alarm_time->rtc_am_pm = (uint32_t)(reg_alrm0td & RTC_ALRM0TD_PM);
  342. rtc_alarm_time->rtc_weekday_or_date = (uint32_t)(reg_alrm0td & RTC_ALRM0TD_DOWS);
  343. rtc_alarm_time->rtc_alarm_day = (uint8_t)GET_ALRM0TD_DAY(reg_alrm0td);
  344. rtc_alarm_time->rtc_alarm_hour = (uint8_t)GET_ALRM0TD_HR(reg_alrm0td);
  345. rtc_alarm_time->rtc_alarm_minute = (uint8_t)GET_ALRM0TD_MN(reg_alrm0td);
  346. rtc_alarm_time->rtc_alarm_second = (uint8_t)GET_ALRM0TD_SC(reg_alrm0td);
  347. }
  348. /*!
  349. \brief get RTC alarm subsecond
  350. \param[in] none
  351. \param[out] none
  352. \retval RTC alarm subsecond value
  353. */
  354. uint32_t rtc_alarm_subsecond_get(void)
  355. {
  356. return ((uint32_t)(RTC_ALRM0SS & RTC_ALRM0SS_SSC));
  357. }
  358. /*!
  359. \brief enable RTC alarm
  360. \param[in] none
  361. \param[out] none
  362. \retval none
  363. */
  364. void rtc_alarm_enable(void)
  365. {
  366. /* disable the write protection */
  367. RTC_WPK = RTC_UNLOCK_KEY1;
  368. RTC_WPK = RTC_UNLOCK_KEY2;
  369. RTC_CTL |= RTC_CTL_ALRM0EN;
  370. /* enable the write protection */
  371. RTC_WPK = RTC_LOCK_KEY;
  372. }
  373. /*!
  374. \brief disable RTC alarm
  375. \param[in] none
  376. \param[out] none
  377. \retval ErrStatus: ERROR or SUCCESS
  378. */
  379. ErrStatus rtc_alarm_disable(void)
  380. {
  381. volatile uint32_t time_index = RTC_ALRM0WF_TIMEOUT;
  382. ErrStatus error_status = ERROR;
  383. uint32_t flag_status = RESET;
  384. /* disable the write protection */
  385. RTC_WPK = RTC_UNLOCK_KEY1;
  386. RTC_WPK = RTC_UNLOCK_KEY2;
  387. /* clear the state of alarm */
  388. RTC_CTL &= (uint32_t)(~RTC_CTL_ALRM0EN);
  389. /* wait until ALRM0WF flag to be set after the alarm is disabled */
  390. do{
  391. flag_status = RTC_STAT & RTC_STAT_ALRM0WF;
  392. }while((--time_index > 0x00U) && (RESET == flag_status));
  393. if(RESET != flag_status){
  394. error_status = SUCCESS;
  395. }
  396. /* enable the write protection */
  397. RTC_WPK = RTC_LOCK_KEY;
  398. return error_status;
  399. }
  400. /*!
  401. \brief enable RTC time-stamp
  402. \param[in] edge: specify which edge to detect of time-stamp
  403. only one parameter can be selected which is shown as below:
  404. \arg RTC_TIMESTAMP_RISING_EDGE: rising edge is valid event edge for timestamp event
  405. \arg RTC_TIMESTAMP_FALLING_EDGE: falling edge is valid event edge for timestamp event
  406. \param[out] none
  407. \retval none
  408. */
  409. void rtc_timestamp_enable(uint32_t edge)
  410. {
  411. uint32_t reg_ctl = 0x00U;
  412. /* clear the bits to be configured in RTC_CTL */
  413. reg_ctl = (uint32_t)(RTC_CTL & (uint32_t)(~(RTC_CTL_TSEG | RTC_CTL_TSEN)));
  414. /* new configuration */
  415. reg_ctl |= (uint32_t)(edge | RTC_CTL_TSEN);
  416. /* disable the write protection */
  417. RTC_WPK = RTC_UNLOCK_KEY1;
  418. RTC_WPK = RTC_UNLOCK_KEY2;
  419. RTC_CTL = (uint32_t)reg_ctl;
  420. /* enable the write protection */
  421. RTC_WPK = RTC_LOCK_KEY;
  422. }
  423. /*!
  424. \brief disable RTC time-stamp
  425. \param[in] none
  426. \param[out] none
  427. \retval none
  428. */
  429. void rtc_timestamp_disable(void)
  430. {
  431. /* disable the write protection */
  432. RTC_WPK = RTC_UNLOCK_KEY1;
  433. RTC_WPK = RTC_UNLOCK_KEY2;
  434. /* clear the TSEN bit */
  435. RTC_CTL &= (uint32_t)(~ RTC_CTL_TSEN);
  436. /* enable the write protection */
  437. RTC_WPK = RTC_LOCK_KEY;
  438. }
  439. /*!
  440. \brief get RTC timestamp time and date
  441. \param[in] none
  442. \param[out] rtc_timestamp: pointer to a rtc_timestamp_struct structure which contains
  443. parameters for RTC time-stamp configuration
  444. members of the structure and the member values are shown as below:
  445. rtc_timestamp_month: RTC_JAN, RTC_FEB, RTC_MAR, RTC_APR, RTC_MAY, RTC_JUN,
  446. RTC_JUL, RTC_AUG, RTC_SEP, RTC_OCT, RTC_NOV, RTC_DEC
  447. rtc_timestamp_date: 0x1 - 0x31(BCD format)
  448. rtc_timestamp_day: RTC_MONDAY, RTC_TUESDAY, RTC_WEDSDAY, RTC_THURSDAY, RTC_FRIDAY,
  449. RTC_SATURDAY, RTC_SUNDAY if RTC_ALARM_WEEKDAY_SELECTED is set
  450. rtc_timestamp_hour: 0x0 - 0x12(BCD format) or 0x0 - 0x23(BCD format) depending on the rtc_display_format
  451. rtc_timestamp_minute: 0x0 - 0x59(BCD format)
  452. rtc_timestamp_second: 0x0 - 0x59(BCD format)
  453. rtc_am_pm: RTC_AM, RTC_PM
  454. \retval none
  455. */
  456. void rtc_timestamp_get(rtc_timestamp_struct* rtc_timestamp)
  457. {
  458. uint32_t temp_tts = 0x00U, temp_dts = 0x00U;
  459. /* get the value of time_stamp registers */
  460. temp_tts = (uint32_t)RTC_TTS;
  461. temp_dts = (uint32_t)RTC_DTS;
  462. /* get timestamp time and construct the rtc_timestamp_struct structure */
  463. rtc_timestamp->rtc_am_pm = (uint32_t)(temp_tts & RTC_TTS_PM);
  464. rtc_timestamp->rtc_timestamp_month = (uint8_t)GET_DTS_MON(temp_dts);
  465. rtc_timestamp->rtc_timestamp_date = (uint8_t)GET_DTS_DAY(temp_dts);
  466. rtc_timestamp->rtc_timestamp_day = (uint8_t)GET_DTS_DOW(temp_dts);
  467. rtc_timestamp->rtc_timestamp_hour = (uint8_t)GET_TTS_HR(temp_tts);
  468. rtc_timestamp->rtc_timestamp_minute = (uint8_t)GET_TTS_MN(temp_tts);
  469. rtc_timestamp->rtc_timestamp_second = (uint8_t)GET_TTS_SC(temp_tts);
  470. }
  471. /*!
  472. \brief get RTC time-stamp subsecond
  473. \param[in] none
  474. \param[out] none
  475. \retval RTC time-stamp subsecond value
  476. */
  477. uint32_t rtc_timestamp_subsecond_get(void)
  478. {
  479. return ((uint32_t)RTC_SSTS);
  480. }
  481. /*!
  482. \brief enable RTC tamper
  483. \param[in] rtc_tamper: pointer to a rtc_tamper_struct structure which contains
  484. parameters for RTC tamper configuration
  485. members of the structure and the member values are shown as below:
  486. rtc_tamper_source: RTC_TAMPER0, RTC_TAMPER1
  487. rtc_tamper_trigger: RTC_TAMPER_TRIGGER_EDGE_RISING, RTC_TAMPER_TRIGGER_EDGE_FALLING
  488. RTC_TAMPER_TRIGGER_LEVEL_LOW, RTC_TAMPER_TRIGGER_LEVEL_HIGH
  489. rtc_tamper_filter: RTC_FLT_EDGE, RTC_FLT_2S, RTC_FLT_4S, RTC_FLT_8S
  490. rtc_tamper_sample_frequency: RTC_FREQ_DIV32768, RTC_FREQ_DIV16384, RTC_FREQ_DIV8192,
  491. RTC_FREQ_DIV4096, RTC_FREQ_DIV2048, RTC_FREQ_DIV1024,
  492. RTC_FREQ_DIV512, RTC_FREQ_DIV256
  493. rtc_tamper_precharge_enable: DISABLE, ENABLE
  494. rtc_tamper_precharge_time: RTC_PRCH_1C, RTC_PRCH_2C, RTC_PRCH_4C, RTC_PRCH_8C
  495. rtc_tamper_with_timestamp: DISABLE, ENABLE
  496. \param[out] none
  497. \retval none
  498. */
  499. void rtc_tamper_enable(rtc_tamper_struct* rtc_tamper)
  500. {
  501. /* disable tamper */
  502. RTC_TAMP &= (uint32_t)~(rtc_tamper->rtc_tamper_source);
  503. /* tamper filter must be used when the tamper source is voltage level detection */
  504. RTC_TAMP &= (uint32_t)~RTC_TAMP_FLT;
  505. /* the tamper source is voltage level detection */
  506. if(rtc_tamper->rtc_tamper_filter != RTC_FLT_EDGE ){
  507. RTC_TAMP &= (uint32_t)~(RTC_TAMP_DISPU | RTC_TAMP_PRCH | RTC_TAMP_FREQ | RTC_TAMP_FLT);
  508. /* check if the tamper pin need precharge, if need, then configure the precharge time */
  509. if(DISABLE == rtc_tamper->rtc_tamper_precharge_enable){
  510. RTC_TAMP |= (uint32_t)RTC_TAMP_DISPU;
  511. }else{
  512. RTC_TAMP |= (uint32_t)(rtc_tamper->rtc_tamper_precharge_time);
  513. }
  514. RTC_TAMP |= (uint32_t)(rtc_tamper->rtc_tamper_sample_frequency);
  515. RTC_TAMP |= (uint32_t)(rtc_tamper->rtc_tamper_filter);
  516. }
  517. RTC_TAMP &= (uint32_t)~RTC_TAMP_TPTS;
  518. if(DISABLE != rtc_tamper->rtc_tamper_with_timestamp){
  519. /* the tamper event also cause a time-stamp event */
  520. RTC_TAMP |= (uint32_t)RTC_TAMP_TPTS;
  521. }
  522. /* configure the tamper trigger */
  523. RTC_TAMP &= ((uint32_t)~((rtc_tamper->rtc_tamper_source) << RTC_TAMPER_TRIGGER_POS));
  524. if(RTC_TAMPER_TRIGGER_EDGE_RISING != rtc_tamper->rtc_tamper_trigger){
  525. RTC_TAMP |= (uint32_t)((rtc_tamper->rtc_tamper_source)<< RTC_TAMPER_TRIGGER_POS);
  526. }
  527. /* enable tamper */
  528. RTC_TAMP |= (uint32_t)(rtc_tamper->rtc_tamper_source);
  529. }
  530. /*!
  531. \brief disable RTC tamper
  532. \param[in] source: specify which tamper source to be disabled
  533. only one parameter can be selected which is shown as below:
  534. \arg RTC_TAMPER0
  535. \arg RTC_TAMPER1
  536. \param[out] none
  537. \retval none
  538. */
  539. void rtc_tamper_disable(uint32_t source)
  540. {
  541. /* disable tamper */
  542. RTC_TAMP &= (uint32_t)~source;
  543. }
  544. /*!
  545. \brief enable specified RTC interrupt
  546. \param[in] interrupt: specify which interrupt source to be enabled
  547. only one parameter can be selected which is shown as below:
  548. \arg RTC_INT_TIMESTAMP: timestamp interrupt
  549. \arg RTC_INT_ALARM: alarm interrupt
  550. \arg RTC_INT_TAMP: tamp interrupt
  551. \param[out] none
  552. \retval none
  553. */
  554. void rtc_interrupt_enable(uint32_t interrupt)
  555. {
  556. /* disable the write protection */
  557. RTC_WPK = RTC_UNLOCK_KEY1;
  558. RTC_WPK = RTC_UNLOCK_KEY2;
  559. /* enable the interrupts in RTC_CTL register */
  560. RTC_CTL |= (uint32_t)(interrupt & (uint32_t)~RTC_TAMP_TPIE);
  561. /* enable the interrupts in RTC_TAMP register */
  562. RTC_TAMP |= (uint32_t)(interrupt & RTC_TAMP_TPIE);
  563. /* enable the write protection */
  564. RTC_WPK = RTC_LOCK_KEY;
  565. }
  566. /*!
  567. \brief disble specified RTC interrupt
  568. \param[in] interrupt: specify which interrupt source to be disabled
  569. only one parameter can be selected which is shown as below:
  570. \arg RTC_INT_TIMESTAMP: timestamp interrupt
  571. \arg RTC_INT_ALARM: alarm interrupt
  572. \arg RTC_INT_TAMP: tamp interrupt
  573. \param[out] none
  574. \retval none
  575. */
  576. void rtc_interrupt_disable(uint32_t interrupt)
  577. {
  578. /* disable the write protection */
  579. RTC_WPK = RTC_UNLOCK_KEY1;
  580. RTC_WPK = RTC_UNLOCK_KEY2;
  581. /* disable the interrupts in RTC_CTL register */
  582. RTC_CTL &= (uint32_t)~(interrupt & (uint32_t)~RTC_TAMP_TPIE);
  583. /* disable the interrupts in RTC_TAMP register */
  584. RTC_TAMP &= (uint32_t)~(interrupt & RTC_TAMP_TPIE);
  585. /* enable the write protection */
  586. RTC_WPK = RTC_LOCK_KEY;
  587. }
  588. /*!
  589. \brief check specified flag
  590. \param[in] flag: specify which flag to check
  591. only one parameter can be selected which is shown as below:
  592. \arg RTC_FLAG_RECALIBRATION: recalibration pending flag
  593. \arg RTC_FLAG_TAMP1: tamper 1 event flag
  594. \arg RTC_FLAG_TAMP0: tamper 0 event flag
  595. \arg RTC_FLAG_TIMESTAMP_OVERFLOW: time-stamp overflow event flag
  596. \arg RTC_FLAG_TIMESTAMP: time-stamp event flag
  597. \arg RTC_FLAG_ALARM0: alarm event flag
  598. \arg RTC_FLAG_INIT: init mode event flag
  599. \arg RTC_FLAG_RSYN: time and date registers synchronized event flag
  600. \arg RTC_FLAG_YCM: year parameter configured event flag
  601. \arg RTC_FLAG_SHIFT: shift operation pending flag
  602. \arg RTC_FLAG_ALARM0_WRITTEN: alarm writen available flag
  603. \param[out] none
  604. \retval FlagStatus: SET or RESET
  605. */
  606. FlagStatus rtc_flag_get(uint32_t flag)
  607. {
  608. FlagStatus flag_state = RESET;
  609. if(RESET != (RTC_STAT & flag)){
  610. flag_state = SET;
  611. }
  612. return flag_state;
  613. }
  614. /*!
  615. \brief clear specified flag
  616. \param[in] flag: specify which flag to clear
  617. \arg RTC_FLAG_TAMP1: tamper 1 event flag
  618. \arg RTC_FLAG_TAMP0: tamper 0 event flag
  619. \arg RTC_FLAG_TIMESTAMP_OVERFLOW: time-stamp overflow event flag
  620. \arg RTC_FLAG_TIMESTAMP: time-stamp event flag
  621. \arg RTC_FLAG_ALARM0: alarm event flag
  622. \arg RTC_FLAG_RSYN: time and date registers synchronized event flag
  623. \param[out] none
  624. \retval none
  625. */
  626. void rtc_flag_clear(uint32_t flag)
  627. {
  628. RTC_STAT &= (uint32_t)(~flag);
  629. }
  630. /*!
  631. \brief configure rtc alternate output source
  632. \param[in] source: specify signal to output
  633. only one parameter can be selected which is shown as below:
  634. \arg RTC_CALIBRATION_512HZ: when the LSE freqency is 32768Hz and the RTC_PSC
  635. is the default value, output 512Hz signal
  636. \arg RTC_CALIBRATION_1HZ: when the LSE freqency is 32768Hz and the RTC_PSC
  637. is the default value, output 1Hz signal
  638. \arg RTC_ALARM_HIGH: when the alarm flag is set, the output pin is high
  639. \arg RTC_ALARM_LOW: when the Alarm flag is set, the output pin is low
  640. \param[in] mode: specify the output pin (PC13) mode when output alarm signal
  641. only one parameter can be selected which is shown as below:
  642. \arg RTC_ALARM_OUTPUT_OD: open drain mode
  643. \arg RTC_ALARM_OUTPUT_PP: push pull mode
  644. \param[out] none
  645. \retval none
  646. */
  647. void rtc_alter_output_config(uint32_t source, uint32_t mode)
  648. {
  649. /* disable the write protection */
  650. RTC_WPK = RTC_UNLOCK_KEY1;
  651. RTC_WPK = RTC_UNLOCK_KEY2;
  652. RTC_CTL &= (uint32_t)~(RTC_CTL_COEN | RTC_CTL_OS | RTC_CTL_OPOL | RTC_CTL_COS);
  653. RTC_CTL |= (uint32_t)(source);
  654. /* alarm output */
  655. if(RESET != (source & RTC_OS_ENABLE)){
  656. RTC_TAMP &= (uint32_t)~(RTC_TAMP_PC13VAL);
  657. RTC_TAMP |= (uint32_t)(mode);
  658. }
  659. /* enable the write protection */
  660. RTC_WPK = RTC_LOCK_KEY;
  661. }
  662. /*!
  663. \brief configure RTC calibration register
  664. \param[in] window: select calibration window
  665. only one parameter can be selected which is shown as below:
  666. \arg RTC_CALIBRATION_WINDOW_32S: 2exp20 RTCCLK cycles, 32s if RTCCLK = 32768 Hz
  667. \arg RTC_CALIBRATION_WINDOW_16S: 2exp19 RTCCLK cycles, 16s if RTCCLK = 32768 Hz
  668. \arg RTC_CALIBRATION_WINDOW_8S: 2exp18 RTCCLK cycles, 8s if RTCCLK = 32768 Hz
  669. \param[in] plus: add RTC clock or not
  670. only one parameter can be selected which is shown as below:
  671. \arg RTC_CALIBRATION_PLUS_SET: add one RTC clock every 2048 rtc clock
  672. \arg RTC_CALIBRATION_PLUS_RESET: no effect
  673. \param[in] minus: the RTC clock to minus during the calibration window(0x0 - 0x1FF)
  674. \param[out] none
  675. \retval ErrStatus: ERROR or SUCCESS
  676. */
  677. ErrStatus rtc_calibration_config(uint32_t window, uint32_t plus, uint32_t minus)
  678. {
  679. uint32_t time_index = RTC_HRFC_TIMEOUT;
  680. ErrStatus error_status = ERROR;
  681. uint32_t flag_status = RESET;
  682. /* disable the write protection */
  683. RTC_WPK = RTC_UNLOCK_KEY1;
  684. RTC_WPK = RTC_UNLOCK_KEY2;
  685. /* check if a calibration operation is ongoing */
  686. do{
  687. flag_status = RTC_STAT & RTC_STAT_SCPF;
  688. }while((--time_index > 0x00U) && (RESET != flag_status));
  689. if(RESET == flag_status){
  690. RTC_HRFC = (uint32_t)(window | plus | HRFC_CMSK(minus));
  691. error_status = SUCCESS;
  692. }
  693. /* enable the write protection */
  694. RTC_WPK = RTC_LOCK_KEY;
  695. return error_status;
  696. }
  697. /*!
  698. \brief ajust the daylight saving time by adding or substracting one hour from the current time
  699. \param[in] operation: hour ajustment operation
  700. only one parameter can be selected which is shown as below:
  701. \arg RTC_CTL_A1H: add one hour
  702. \arg RTC_CTL_S1H: substract one hour
  703. \param[out] none
  704. \retval none
  705. */
  706. void rtc_hour_adjust(uint32_t operation)
  707. {
  708. /* disable the write protection */
  709. RTC_WPK = RTC_UNLOCK_KEY1;
  710. RTC_WPK = RTC_UNLOCK_KEY2;
  711. RTC_CTL |= (uint32_t)(operation);
  712. /* enable the write protection */
  713. RTC_WPK = RTC_LOCK_KEY;
  714. }
  715. /*!
  716. \brief ajust RTC second or subsecond value of current time
  717. \param[in] add: add 1s to current time or not
  718. only one parameter can be selected which is shown as below:
  719. \arg RTC_SHIFT_ADD1S_RESET: no effect
  720. \arg RTC_SHIFT_ADD1S_SET: add 1s to current time
  721. \param[in] minus: number of subsecond to minus from current time(0x0 - 0x7FFF)
  722. \param[out] none
  723. \retval ErrStatus: ERROR or SUCCESS
  724. */
  725. ErrStatus rtc_second_adjust(uint32_t add, uint32_t minus)
  726. {
  727. uint32_t time_index = RTC_SHIFTCTL_TIMEOUT;
  728. ErrStatus error_status = ERROR;
  729. uint32_t flag_status = RESET;
  730. uint32_t temp=0U;
  731. /* disable the write protection */
  732. RTC_WPK = RTC_UNLOCK_KEY1;
  733. RTC_WPK = RTC_UNLOCK_KEY2;
  734. /* check if a shift operation is ongoing */
  735. do{
  736. flag_status = RTC_STAT & RTC_STAT_SOPF;
  737. }while((--time_index > 0x00U) && (RESET != flag_status));
  738. temp = RTC_CTL & RTC_CTL_REFEN;
  739. /* check if the function of reference clock detection is disabled */
  740. if((RESET == flag_status) && (RESET == temp)){
  741. RTC_SHIFTCTL = (uint32_t)(add | SHIFTCTL_SFS(minus));
  742. error_status = rtc_register_sync_wait();
  743. }
  744. /* enable the write protection */
  745. RTC_WPK = RTC_LOCK_KEY;
  746. return error_status;
  747. }
  748. /*!
  749. \brief enable RTC bypass shadow registers function
  750. \param[in] none
  751. \param[out] none
  752. \retval none
  753. */
  754. void rtc_bypass_shadow_enable(void)
  755. {
  756. /* disable the write protection */
  757. RTC_WPK = RTC_UNLOCK_KEY1;
  758. RTC_WPK = RTC_UNLOCK_KEY2;
  759. RTC_CTL |= (uint8_t)RTC_CTL_BPSHAD;
  760. /* enable the write protection */
  761. RTC_WPK = RTC_LOCK_KEY;
  762. }
  763. /*!
  764. \brief disable RTC bypass shadow registers function
  765. \param[in] none
  766. \param[out] none
  767. \retval none
  768. */
  769. void rtc_bypass_shadow_disable(void)
  770. {
  771. /* disable the write protection */
  772. RTC_WPK = RTC_UNLOCK_KEY1;
  773. RTC_WPK = RTC_UNLOCK_KEY2;
  774. RTC_CTL &= (uint8_t)~RTC_CTL_BPSHAD;
  775. /* enable the write protection */
  776. RTC_WPK = RTC_LOCK_KEY;
  777. }
  778. /*!
  779. \brief enable RTC reference clock detection function
  780. \param[in] none
  781. \param[out] none
  782. \retval ErrStatus: ERROR or SUCCESS
  783. */
  784. ErrStatus rtc_refclock_detection_enable(void)
  785. {
  786. ErrStatus error_status = ERROR;
  787. /* disable the write protection */
  788. RTC_WPK = RTC_UNLOCK_KEY1;
  789. RTC_WPK = RTC_UNLOCK_KEY2;
  790. /* enter init mode */
  791. error_status = rtc_init_mode_enter();
  792. if(ERROR != error_status){
  793. RTC_CTL |= (uint32_t)RTC_CTL_REFEN;
  794. /* exit init mode */
  795. rtc_init_mode_exit();
  796. }
  797. /* enable the write protection */
  798. RTC_WPK = RTC_LOCK_KEY;
  799. return error_status;
  800. }
  801. /*!
  802. \brief disable RTC reference clock detection function
  803. \param[in] none
  804. \param[out] none
  805. \retval ErrStatus: ERROR or SUCCESS
  806. */
  807. ErrStatus rtc_refclock_detection_disable(void)
  808. {
  809. ErrStatus error_status = ERROR;
  810. /* disable the write protection */
  811. RTC_WPK = RTC_UNLOCK_KEY1;
  812. RTC_WPK = RTC_UNLOCK_KEY2;
  813. /* enter init mode */
  814. error_status = rtc_init_mode_enter();
  815. if(ERROR != error_status){
  816. RTC_CTL &= (uint32_t)~RTC_CTL_REFEN;
  817. /* exit init mode */
  818. rtc_init_mode_exit();
  819. }
  820. /* enable the write protection */
  821. RTC_WPK = RTC_LOCK_KEY;
  822. return error_status;
  823. }