nu_rtc.c 31 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153
  1. /**************************************************************************//**
  2. * @file RTC.c
  3. * @brief N9H30 RTC driver source file
  4. *
  5. * @note
  6. * SPDX-License-Identifier: Apache-2.0
  7. * Copyright (C) 2018 Nuvoton Technology Corp. All rights reserved.
  8. *****************************************************************************/
  9. #include <stdio.h>
  10. #include <stdlib.h>
  11. #include <string.h>
  12. #include "N9H30.h"
  13. #include "nu_sys.h"
  14. #include "nu_rtc.h"
  15. /** @addtogroup N9H30_Device_Driver N9H30 Device Driver
  16. @{
  17. */
  18. /** @addtogroup N9H30_RTC_Driver RTC Driver
  19. @{
  20. */
  21. /** @addtogroup N9H30_RTC_EXPORTED_FUNCTIONS RTC Exported Functions
  22. @{
  23. */
  24. /// @cond HIDDEN_SYMBOLS
  25. static CHAR g_chHourMode = 0;
  26. static BOOL volatile g_bIsEnableTickInt = FALSE;
  27. static BOOL volatile g_bIsEnableAlarmInt = FALSE;
  28. static UINT32 volatile g_u32Reg, g_u32Reg1, g_u32hiYear, g_u32loYear, g_u32hiMonth, g_u32loMonth, g_u32hiDay, g_u32loDay;
  29. static UINT32 volatile g_u32hiHour, g_u32loHour, g_u32hiMin, g_u32loMin, g_u32hiSec, g_u32loSec;
  30. UINT32 volatile i, Wait;
  31. VOID RTC_Check(void)
  32. {
  33. i = 0;
  34. Wait = inp32(REG_RTC_INTSTS) & RTC_INTSTS_REGWRBUSY_Msk;
  35. while (Wait == RTC_INTSTS_REGWRBUSY_Msk)
  36. {
  37. Wait = inp32(REG_RTC_INTSTS) & RTC_INTSTS_REGWRBUSY_Msk;
  38. i++;
  39. if (i > RTC_WAIT_COUNT)
  40. {
  41. break;
  42. }
  43. }
  44. }
  45. /// @endcond HIDDEN_SYMBOLS
  46. /**
  47. * @brief Set 32k Frequency Compensation Data
  48. *
  49. * @param[in] i32FrequencyX100 Specify the RTC clock X100, ex: 3277365 means 32773.65.
  50. *
  51. * @return E_RTC_ERR_FCR_VALUE Wrong Compensation VALUE
  52. * E_RTC_SUCCESS Success
  53. *
  54. * @details This API is used to compensate the 32 kHz frequency by current LXT frequency for RTC application.
  55. */
  56. UINT32 RTC_DoFrequencyCompensation(INT32 i32FrequencyX100)
  57. {
  58. INT32 i32RegInt, i32RegFra;
  59. UINT32 u32Reg;
  60. /* Compute integer and fraction for RTC FCR register */
  61. i32RegInt = (i32FrequencyX100 / 100) - RTC_FCR_REFERENCE;
  62. i32RegFra = (((i32FrequencyX100 % 100)) * 60) / 100;
  63. /* Judge Integer part is reasonable */
  64. if ((i32RegInt < 0) | (i32RegInt > 15))
  65. {
  66. return E_RTC_ERR_FCR_VALUE;
  67. }
  68. u32Reg = (uint32_t)((i32RegInt << 8) | i32RegFra);
  69. RTC_WriteEnable(1);
  70. outp32(REG_RTC_FREQADJ, u32Reg);
  71. RTC_Check();
  72. return E_RTC_SUCCESS;
  73. }
  74. /**
  75. * @brief RTC access register enable
  76. *
  77. * @param[in] bEnable 1: Enable access register
  78. * 0: Disable access register
  79. *
  80. * @retval E_RTC_ERR_EIO Time-out error
  81. * @retval E_RTC_SUCCESS Success
  82. *
  83. */
  84. UINT32 RTC_WriteEnable(BOOL bEnable)
  85. {
  86. INT32 volatile i32i;
  87. RTC_Check();
  88. if (bEnable)
  89. {
  90. outp32(REG_RTC_RWEN, RTC_WRITE_KEY);
  91. RTC_Check();
  92. for (i32i = 0 ; i32i < RTC_WAIT_COUNT ; i32i++)
  93. {
  94. /*-------------------------------------------------------------------------------------------------*/
  95. /* check RTC_RWEN[16] to find out RTC write enable */
  96. /*-------------------------------------------------------------------------------------------------*/
  97. if (inp32(REG_RTC_RWEN) & 0x10000)
  98. {
  99. break;
  100. }
  101. }
  102. if (i32i == RTC_WAIT_COUNT)
  103. {
  104. //sysprintf ("\nRTC: 3, set write enable FAILED!\n");
  105. return E_RTC_ERR_EIO;
  106. }
  107. }
  108. else
  109. {
  110. for (i32i = 0 ; i32i < RTC_WAIT_COUNT ; i32i++)
  111. {
  112. if (inp32(REG_RTC_RWEN) == 0)
  113. {
  114. break;
  115. }
  116. }
  117. }
  118. return E_RTC_SUCCESS;
  119. }
  120. /**
  121. * @brief Initial RTC and install ISR
  122. * @retval E_RTC_ERR_EIO Initial RTC time-out
  123. * @retval E_RTC_SUCCESS Success
  124. *
  125. */
  126. UINT32 RTC_Init(void)
  127. {
  128. INT32 i32i;
  129. /*-----------------------------------------------------------------------------------------------------*/
  130. /* When RTC is power on, write 0xa5eb1357 to RTC_INIR to reset all logic. */
  131. /*-----------------------------------------------------------------------------------------------------*/
  132. outp32(REG_RTC_INIT, RTC_INIT_KEY);
  133. RTC_Check();
  134. for (i32i = 0 ; i32i < RTC_WAIT_COUNT ; i32i++)
  135. {
  136. if (inp32(REG_RTC_INIT) & 0x01)
  137. {
  138. /* Check RTC_INIR[0] to find out RTC reset signal */
  139. break;
  140. }
  141. }
  142. if (i32i == RTC_WAIT_COUNT)
  143. {
  144. return E_RTC_ERR_EIO;
  145. }
  146. /*-----------------------------------------------------------------------------------------------------*/
  147. /* Install RTC ISR */
  148. /*-----------------------------------------------------------------------------------------------------*/
  149. outp32(REG_RTC_RWEN, RTC_WRITE_KEY);
  150. RTC_Check();
  151. for (i32i = 0 ; i32i < RTC_WAIT_COUNT ; i32i++)
  152. {
  153. /*-------------------------------------------------------------------------------------------------*/
  154. /* check RTC_RWEN[16] to find out RTC write enable */
  155. /*-------------------------------------------------------------------------------------------------*/
  156. if (inp32(REG_RTC_RWEN) & 0x10000)
  157. {
  158. break;
  159. }
  160. }
  161. if (i32i == RTC_WAIT_COUNT)
  162. {
  163. return E_RTC_ERR_EIO;
  164. }
  165. return E_RTC_SUCCESS;
  166. }
  167. /**
  168. * @brief Set Current Timer
  169. *
  170. * @param[in] *sPt Specify the time property and current time. It includes:
  171. * - u8cClockDisplay: \ref RTC_CLOCK_12 / \ref RTC_CLOCK_24
  172. * - u8cAmPm: \ref RTC_AM / \ref RTC_PM
  173. * - u32cSecond: Second value
  174. * - u32cMinute: Minute value
  175. * - u32cHour: Hour value
  176. * - u32cDayOfWeek: Day of week
  177. * - u32cDay: Day value
  178. * - u32cMonth: Month value
  179. * - u32Year: Year value
  180. * - u32AlarmMaskSecond: Mask second alarm
  181. * - u32AlarmMaskMinute: Mask minute alarm
  182. * - u32AlarmMaskHour: Mask hour alarm
  183. * - *pfnAlarmCallBack: Call back function
  184. *
  185. * @retval E_RTC_ERR_EIO Initial RTC time-out
  186. * @retval E_RTC_SUCCESS Success
  187. *
  188. */
  189. UINT32 RTC_Open(S_RTC_TIME_DATA_T *sPt)
  190. {
  191. UINT32 volatile u32Reg;
  192. /*-----------------------------------------------------------------------------------------------------*/
  193. /* DO BASIC JUDGEMENT TO Check RTC time data value is reasonable or not. */
  194. /*-----------------------------------------------------------------------------------------------------*/
  195. if (((sPt->u32Year - RTC_YEAR2000) > 99) |
  196. ((sPt->u32cMonth == 0) || (sPt->u32cMonth > 12)) |
  197. ((sPt->u32cDay == 0) || (sPt->u32cDay > 31)))
  198. {
  199. return E_RTC_ERR_CALENDAR_VALUE;
  200. }
  201. if (sPt->u8cClockDisplay == RTC_CLOCK_12)
  202. {
  203. if ((sPt->u32cHour == 0) || (sPt->u32cHour > 12))
  204. {
  205. return E_RTC_ERR_TIMESACLE_VALUE ;
  206. }
  207. }
  208. else if (sPt->u8cClockDisplay == RTC_CLOCK_24)
  209. {
  210. if (sPt->u32cHour > 23)
  211. {
  212. return E_RTC_ERR_TIMESACLE_VALUE ;
  213. }
  214. }
  215. else
  216. {
  217. return E_RTC_ERR_TIMESACLE_VALUE ;
  218. }
  219. if ((sPt->u32cMinute > 59) |
  220. (sPt->u32cSecond > 59) |
  221. (sPt->u32cSecond > 59))
  222. {
  223. return E_RTC_ERR_TIME_VALUE ;
  224. }
  225. if (sPt->u32cDayOfWeek > 6)
  226. {
  227. return E_RTC_ERR_DWR_VALUE ;
  228. }
  229. /*-----------------------------------------------------------------------------------------------------*/
  230. /* Second, set RTC time data. */
  231. /*-----------------------------------------------------------------------------------------------------*/
  232. if (sPt->u8cClockDisplay == RTC_CLOCK_12)
  233. {
  234. g_chHourMode = RTC_CLOCK_12;
  235. RTC_WriteEnable(1);
  236. outp32(REG_RTC_TIMEFMT, RTC_CLOCK_12);
  237. RTC_Check();
  238. /*-------------------------------------------------------------------------------------------------*/
  239. /* important, range of 12-hour PM mode is 21 upto 32 */
  240. /*-------------------------------------------------------------------------------------------------*/
  241. if (sPt->u8cAmPm == RTC_PM)
  242. sPt->u32cHour += 20;
  243. }
  244. else /* RTC_CLOCK_24 */
  245. {
  246. g_chHourMode = RTC_CLOCK_24;
  247. RTC_WriteEnable(1);
  248. outp32(REG_RTC_TIMEFMT, RTC_CLOCK_24);
  249. RTC_Check();
  250. }
  251. g_u32hiHour = sPt->u32cHour / 10;
  252. g_u32loHour = sPt->u32cHour % 10;
  253. g_u32hiMin = sPt->u32cMinute / 10;
  254. g_u32loMin = sPt->u32cMinute % 10;
  255. g_u32hiSec = sPt->u32cSecond / 10;
  256. g_u32loSec = sPt->u32cSecond % 10;
  257. u32Reg = (g_u32hiHour << 20);
  258. u32Reg |= (g_u32loHour << 16);
  259. u32Reg |= (g_u32hiMin << 12);
  260. u32Reg |= (g_u32loMin << 8);
  261. u32Reg |= (g_u32hiSec << 4);
  262. u32Reg |= g_u32loSec;
  263. g_u32Reg = u32Reg;
  264. RTC_WriteEnable(1);
  265. outp32(REG_RTC_TIME, (UINT32)g_u32Reg);
  266. RTC_Check();
  267. if (sPt->u8cClockDisplay == RTC_CLOCK_12)
  268. {
  269. if (sPt->u8cAmPm == RTC_PM)
  270. sPt->u32cHour -= 20;
  271. }
  272. g_u32hiYear = (sPt->u32Year - RTC_YEAR2000) / 10;
  273. g_u32loYear = (sPt->u32Year - RTC_YEAR2000) % 10;
  274. g_u32hiMonth = sPt->u32cMonth / 10;
  275. g_u32loMonth = sPt->u32cMonth % 10;
  276. g_u32hiDay = sPt->u32cDay / 10;
  277. g_u32loDay = sPt->u32cDay % 10;
  278. u32Reg = (g_u32hiYear << 20);
  279. u32Reg |= (g_u32loYear << 16);
  280. u32Reg |= (g_u32hiMonth << 12);
  281. u32Reg |= (g_u32loMonth << 8);
  282. u32Reg |= (g_u32hiDay << 4);
  283. u32Reg |= g_u32loDay;
  284. g_u32Reg = u32Reg;
  285. RTC_WriteEnable(1);
  286. outp32(REG_RTC_CAL, (UINT32)g_u32Reg);
  287. RTC_Check();
  288. RTC_WriteEnable(1);
  289. outp32(REG_RTC_WEEKDAY, (UINT32)sPt->u32cDayOfWeek);
  290. RTC_Check();
  291. return E_RTC_SUCCESS;
  292. }
  293. /**
  294. * @brief Read current date/time or alarm date/time from RTC
  295. *
  296. * @param[in] eTime \ref RTC_CURRENT_TIME / \ref RTC_ALARM_TIME
  297. * @param[out] *sPt Specify the time property and current time. It includes:
  298. * - u8cClockDisplay: \ref RTC_CLOCK_12 / \ref RTC_CLOCK_24
  299. * - u8cAmPm: \ref RTC_AM / \ref RTC_PM
  300. * - u32cSecond: Second value
  301. * - u32cMinute: Minute value
  302. * - u32cHour: Hour value
  303. * - u32cDayOfWeek: Day of week
  304. * - u32cDay: Day value
  305. * - u32cMonth: Month value
  306. * - u32Year: Year value
  307. * - u32AlarmMaskSecond: Mask second alarm
  308. * - u32AlarmMaskMinute: Mask minute alarm
  309. * - u32AlarmMaskHour: Mask hour alarm
  310. * - *pfnAlarmCallBack: Call back function
  311. *
  312. * @retval E_RTC_ERR_ENOTTY Wrong select time
  313. * @retval E_RTC_SUCCESS Success
  314. *
  315. */
  316. UINT32 RTC_Read(E_RTC_TIME_SELECT eTime, S_RTC_TIME_DATA_T *sPt)
  317. {
  318. UINT32 u32Tmp;
  319. sPt->u8cClockDisplay = inp32(REG_RTC_TIMEFMT); /* 12/24-hour */
  320. sPt->u32cDayOfWeek = inp32(REG_RTC_WEEKDAY); /* Day of week */
  321. switch (eTime)
  322. {
  323. case RTC_CURRENT_TIME:
  324. {
  325. g_u32Reg = inp32(REG_RTC_CAL);
  326. g_u32Reg1 = inp32(REG_RTC_TIME);
  327. break;
  328. }
  329. case RTC_ALARM_TIME:
  330. {
  331. g_u32Reg = inp32(REG_RTC_CALM);
  332. g_u32Reg1 = inp32(REG_RTC_TALM);
  333. break;
  334. }
  335. default:
  336. {
  337. return E_RTC_ERR_ENOTTY;
  338. }
  339. }
  340. g_u32hiYear = (g_u32Reg & 0xF00000) >> 20;
  341. g_u32loYear = (g_u32Reg & 0xF0000) >> 16;
  342. g_u32hiMonth = (g_u32Reg & 0x1000) >> 12;
  343. g_u32loMonth = (g_u32Reg & 0xF00) >> 8;
  344. g_u32hiDay = (g_u32Reg & 0x30) >> 4;
  345. g_u32loDay = g_u32Reg & 0xF;
  346. u32Tmp = (g_u32hiYear * 10);
  347. u32Tmp += g_u32loYear;
  348. sPt->u32Year = u32Tmp + RTC_YEAR2000;
  349. u32Tmp = (g_u32hiMonth * 10);
  350. sPt->u32cMonth = u32Tmp + g_u32loMonth;
  351. u32Tmp = (g_u32hiDay * 10);
  352. sPt->u32cDay = u32Tmp + g_u32loDay;
  353. g_u32hiHour = (g_u32Reg1 & 0x300000) >> 20;
  354. g_u32loHour = (g_u32Reg1 & 0xF0000) >> 16;
  355. g_u32hiMin = (g_u32Reg1 & 0x7000) >> 12;
  356. g_u32loMin = (g_u32Reg1 & 0xF00) >> 8;
  357. g_u32hiSec = (g_u32Reg1 & 0x70) >> 4;
  358. g_u32loSec = g_u32Reg1 & 0xF;
  359. if (sPt->u8cClockDisplay == RTC_CLOCK_12)
  360. {
  361. u32Tmp = (g_u32hiHour * 10);
  362. u32Tmp += g_u32loHour;
  363. sPt->u32cHour = u32Tmp; /* AM: 1~12. PM: 21~32. */
  364. if (eTime == RTC_CURRENT_TIME)
  365. {
  366. if (sPt->u32cHour >= 21)
  367. {
  368. sPt->u8cAmPm = RTC_PM;
  369. sPt->u32cHour -= 20;
  370. }
  371. else
  372. {
  373. sPt->u8cAmPm = RTC_AM;
  374. }
  375. }
  376. else
  377. {
  378. if (sPt->u32cHour < 12)
  379. {
  380. if (sPt->u32cHour == 0)
  381. sPt->u32cHour = 12;
  382. sPt->u8cAmPm = RTC_AM;
  383. }
  384. else
  385. {
  386. sPt->u32cHour -= 12;
  387. sPt->u8cAmPm = RTC_PM;
  388. }
  389. }
  390. u32Tmp = (g_u32hiMin * 10);
  391. u32Tmp += g_u32loMin;
  392. sPt->u32cMinute = u32Tmp;
  393. u32Tmp = (g_u32hiSec * 10);
  394. u32Tmp += g_u32loSec;
  395. sPt->u32cSecond = u32Tmp;
  396. }
  397. else
  398. {
  399. /* RTC_CLOCK_24 */
  400. u32Tmp = (g_u32hiHour * 10);
  401. u32Tmp += g_u32loHour;
  402. sPt->u32cHour = u32Tmp;
  403. u32Tmp = (g_u32hiMin * 10);
  404. u32Tmp += g_u32loMin;
  405. sPt->u32cMinute = u32Tmp;
  406. u32Tmp = (g_u32hiSec * 10);
  407. u32Tmp += g_u32loSec;
  408. sPt->u32cSecond = u32Tmp;
  409. }
  410. return E_RTC_SUCCESS;
  411. }
  412. /**
  413. * @brief Write current date/time or alarm date/time from RTC
  414. *
  415. * @param[in] eTime \ref RTC_CURRENT_TIME / \ref RTC_ALARM_TIME
  416. * @param[in] *sPt Specify the time property and current time. It includes:
  417. * - u8cClockDisplay: \ref RTC_CLOCK_12 / \ref RTC_CLOCK_24
  418. * - u8cAmPm: \ref RTC_AM / \ref RTC_PM
  419. * - u32cSecond: Second value
  420. * - u32cMinute: Minute value
  421. * - u32cHour: Hour value
  422. * - u32cDayOfWeek: Day of week
  423. * - u32cDay: Day value
  424. * - u32cMonth: Month value
  425. * - u32Year: Year value
  426. * - u32AlarmMaskSecond: Mask second alarm
  427. * - u32AlarmMaskMinute: Mask minute alarm
  428. * - u32AlarmMaskHour: Mask hour alarm
  429. * - *pfnAlarmCallBack: Call back function
  430. *
  431. * @retval E_RTC_ERR_ENOTTY Wrong select time
  432. * @retval E_RTC_ERR_CALENDAR_VALUE Wrong calender value
  433. * @retval E_RTC_ERR_TIME_VALUE Wrong time value
  434. * @retval E_RTC_ERR_DWR_VALUE Wrong day of week value
  435. * @retval E_RTC_SUCCESS Success
  436. *
  437. */
  438. UINT32 RTC_Write(E_RTC_TIME_SELECT eTime, S_RTC_TIME_DATA_T *sPt)
  439. {
  440. UINT32 u32Reg;
  441. /*-----------------------------------------------------------------------------------------------------*/
  442. /* Check RTC time data value is reasonable or not. */
  443. /*-----------------------------------------------------------------------------------------------------*/
  444. if (((sPt->u32Year - RTC_YEAR2000) > 99) |
  445. ((sPt->u32cMonth == 0) || (sPt->u32cMonth > 12)) |
  446. ((sPt->u32cDay == 0) || (sPt->u32cDay > 31)))
  447. {
  448. return E_RTC_ERR_CALENDAR_VALUE;
  449. }
  450. if ((sPt->u32Year - RTC_YEAR2000) > 99)
  451. {
  452. return E_RTC_ERR_CALENDAR_VALUE;
  453. }
  454. if ((sPt->u32cMonth == 0) || (sPt->u32cMonth > 12))
  455. {
  456. return E_RTC_ERR_CALENDAR_VALUE;
  457. }
  458. if ((sPt->u32cDay == 0) || (sPt->u32cDay > 31))
  459. {
  460. return E_RTC_ERR_CALENDAR_VALUE;
  461. }
  462. if (sPt->u8cClockDisplay == RTC_CLOCK_12)
  463. {
  464. if ((sPt->u32cHour == 0) || (sPt->u32cHour > 12))
  465. {
  466. return E_RTC_ERR_TIME_VALUE;
  467. }
  468. }
  469. else if (sPt->u8cClockDisplay == RTC_CLOCK_24)
  470. {
  471. if (sPt->u32cHour > 23)
  472. {
  473. return E_RTC_ERR_TIME_VALUE;
  474. }
  475. }
  476. else
  477. {
  478. return E_RTC_ERR_TIME_VALUE;
  479. }
  480. if (sPt->u32cMinute > 59)
  481. {
  482. return E_RTC_ERR_TIME_VALUE;
  483. }
  484. if (sPt->u32cSecond > 59)
  485. {
  486. return E_RTC_ERR_TIME_VALUE;
  487. }
  488. if (sPt->u32cDayOfWeek > 6)
  489. {
  490. return E_RTC_ERR_DWR_VALUE;
  491. }
  492. switch (eTime)
  493. {
  494. case RTC_CURRENT_TIME:
  495. {
  496. /*---------------------------------------------------------------------------------------------*/
  497. /* Second, set RTC time data. */
  498. /*---------------------------------------------------------------------------------------------*/
  499. if (sPt->u8cClockDisplay == RTC_CLOCK_12)
  500. {
  501. g_chHourMode = RTC_CLOCK_12;
  502. RTC_WriteEnable(1);
  503. outp32(REG_RTC_TIMEFMT, RTC_CLOCK_12);
  504. RTC_Check();
  505. /*-----------------------------------------------------------------------------------------*/
  506. /* important, range of 12-hour PM mode is 21 upto 32 */
  507. /*-----------------------------------------------------------------------------------------*/
  508. if (sPt->u8cAmPm == RTC_PM)
  509. {
  510. sPt->u32cHour += 20;
  511. }
  512. }
  513. else /* RTC_CLOCK_24 */
  514. {
  515. g_chHourMode = RTC_CLOCK_24;
  516. RTC_WriteEnable(1);
  517. outp32(REG_RTC_TIMEFMT, RTC_CLOCK_24);
  518. RTC_Check();
  519. }
  520. g_u32hiHour = sPt->u32cHour / 10;
  521. g_u32loHour = sPt->u32cHour % 10;
  522. g_u32hiMin = sPt->u32cMinute / 10;
  523. g_u32loMin = sPt->u32cMinute % 10;
  524. g_u32hiSec = sPt->u32cSecond / 10;
  525. g_u32loSec = sPt->u32cSecond % 10;
  526. u32Reg = (g_u32hiHour << 20);
  527. u32Reg |= (g_u32loHour << 16);
  528. u32Reg |= (g_u32hiMin << 12);
  529. u32Reg |= (g_u32loMin << 8);
  530. u32Reg |= (g_u32hiSec << 4);
  531. u32Reg |= g_u32loSec;
  532. g_u32Reg = u32Reg;
  533. RTC_WriteEnable(1);
  534. outp32(REG_RTC_TIME, (UINT32)g_u32Reg);
  535. RTC_Check();
  536. g_u32hiYear = (sPt->u32Year - RTC_YEAR2000) / 10;
  537. g_u32loYear = (sPt->u32Year - RTC_YEAR2000) % 10;
  538. g_u32hiMonth = sPt->u32cMonth / 10;
  539. g_u32loMonth = sPt->u32cMonth % 10;
  540. g_u32hiDay = sPt->u32cDay / 10;
  541. g_u32loDay = sPt->u32cDay % 10;
  542. u32Reg = (g_u32hiYear << 20);
  543. u32Reg |= (g_u32loYear << 16);
  544. u32Reg |= (g_u32hiMonth << 12);
  545. u32Reg |= (g_u32loMonth << 8);
  546. u32Reg |= (g_u32hiDay << 4);
  547. u32Reg |= g_u32loDay;
  548. g_u32Reg = u32Reg;
  549. RTC_WriteEnable(1);
  550. outp32(REG_RTC_CAL, (UINT32)g_u32Reg);
  551. RTC_Check();
  552. RTC_WriteEnable(1);
  553. outp32(REG_RTC_WEEKDAY, (UINT32) sPt->u32cDayOfWeek);
  554. RTC_Check();
  555. if (sPt->u8cClockDisplay == RTC_CLOCK_12)
  556. {
  557. if (sPt->u8cAmPm == RTC_PM)
  558. {
  559. sPt->u32cHour -= 20;
  560. }
  561. }
  562. return E_RTC_SUCCESS;
  563. }
  564. case RTC_ALARM_TIME:
  565. {
  566. RTC_WriteEnable(1);
  567. outp32(REG_RTC_PWRCTL, inp32(REG_RTC_PWRCTL) & ~RTC_PWRCTL_ALARM_EN_Msk);
  568. RTC_Check();
  569. /*---------------------------------------------------------------------------------------------*/
  570. /* Second, set alarm time data. */
  571. /*---------------------------------------------------------------------------------------------*/
  572. g_u32hiYear = (sPt->u32Year - RTC_YEAR2000) / 10;
  573. g_u32loYear = (sPt->u32Year - RTC_YEAR2000) % 10;
  574. g_u32hiMonth = sPt->u32cMonth / 10;
  575. g_u32loMonth = sPt->u32cMonth % 10;
  576. g_u32hiDay = sPt->u32cDay / 10;
  577. g_u32loDay = sPt->u32cDay % 10;
  578. //u32Reg = ((sPt->u32AlarmMaskDayOfWeek & 0x1) << 31);
  579. u32Reg = ((sPt->u32cDayOfWeek & 0x7) << 24);
  580. //u32Reg|= ((sPt->u32AlarmMaskYear & 0x1) << 30);
  581. u32Reg |= (g_u32hiYear << 20);
  582. u32Reg |= (g_u32loYear << 16);
  583. //u32Reg|= ((sPt->u32AlarmMaskMonth & 0x1) << 29);
  584. u32Reg |= (g_u32hiMonth << 12);
  585. u32Reg |= (g_u32loMonth << 8);
  586. //u32Reg|= ((sPt->u32AlarmMaskDay & 0x1) << 28);
  587. u32Reg |= (g_u32hiDay << 4);
  588. u32Reg |= g_u32loDay;
  589. g_u32Reg = u32Reg;
  590. RTC_WriteEnable(1);
  591. outp32(REG_RTC_CALM, (UINT32)g_u32Reg);
  592. RTC_Check();
  593. if (g_chHourMode == RTC_CLOCK_12)
  594. {
  595. if (sPt->u8cAmPm == RTC_PM) /* important, range of 12-hour PM mode is 21 upto 32 */
  596. {
  597. sPt->u32cHour += 20;
  598. }
  599. }
  600. g_u32hiHour = sPt->u32cHour / 10;
  601. g_u32loHour = sPt->u32cHour % 10;
  602. g_u32hiMin = sPt->u32cMinute / 10;
  603. g_u32loMin = sPt->u32cMinute % 10;
  604. g_u32hiSec = sPt->u32cSecond / 10;
  605. g_u32loSec = sPt->u32cSecond % 10;
  606. u32Reg = ((sPt->u32AlarmMaskHour & 0x1) << 30);
  607. u32Reg |= (g_u32hiHour << 20);
  608. u32Reg |= (g_u32loHour << 16);
  609. u32Reg |= ((sPt->u32AlarmMaskMinute & 0x1) << 29);
  610. u32Reg |= (g_u32hiMin << 12);
  611. u32Reg |= (g_u32loMin << 8);
  612. u32Reg |= ((sPt->u32AlarmMaskSecond & 0x1) << 28);
  613. u32Reg |= (g_u32hiSec << 4);
  614. u32Reg |= g_u32loSec;
  615. g_u32Reg = u32Reg;
  616. RTC_WriteEnable(1);
  617. outp32(REG_RTC_TALM, (UINT32)g_u32Reg);
  618. RTC_Check();
  619. if (sPt->u8cClockDisplay == RTC_CLOCK_12)
  620. {
  621. if (sPt->u8cAmPm == RTC_PM)
  622. {
  623. sPt->u32cHour -= 20;
  624. }
  625. }
  626. /*---------------------------------------------------------------------------------------------*/
  627. /* Finally, enable alarm interrupt. */
  628. /*---------------------------------------------------------------------------------------------*/
  629. RTC_Ioctl(0, RTC_IOC_ENABLE_INT, RTC_ALARM_INT, 0);
  630. RTC_WriteEnable(1);
  631. outp32(REG_RTC_PWRCTL, inp32(REG_RTC_PWRCTL) | RTC_PWRCTL_ALARM_EN_Msk);
  632. RTC_Check();
  633. return E_RTC_SUCCESS;
  634. }
  635. default:
  636. {
  637. return E_RTC_ERR_ENOTTY;
  638. }
  639. }
  640. }
  641. /**
  642. * @brief Support some commands for application.
  643. *
  644. * @param[in] i32Num Interface number. always set 0
  645. * @param[in] eCmd Command
  646. * @param[in] u32Arg0 Arguments for the command
  647. * @param[in] u32Arg1 Arguments for the command.
  648. *
  649. * @retval E_RTC_ERR_ENOTTY Wrong command or argument
  650. * @retval E_RTC_ERR_ENODEV Interface number incorrect
  651. * @retval E_RTC_SUCCESS Success
  652. *
  653. */
  654. UINT32 RTC_Ioctl(INT32 i32Num, E_RTC_CMD eCmd, UINT32 u32Arg0, UINT32 u32Arg1)
  655. {
  656. INT32 i32Ret;
  657. UINT32 u32Reg;
  658. RTC_TICK_T *ptick;
  659. UINT32 u32Tmp;
  660. if (i32Num != 0)
  661. return E_RTC_ERR_ENODEV;
  662. switch (eCmd)
  663. {
  664. case RTC_IOC_IDENTIFY_LEAP_YEAR:
  665. {
  666. u32Reg = inp32(REG_RTC_LEAPYEAR);
  667. if (u32Reg & 0x01)
  668. {
  669. *(PUINT32)u32Arg0 = RTC_LEAP_YEAR;
  670. }
  671. else
  672. {
  673. *(PUINT32)u32Arg0 = 0;
  674. }
  675. break;
  676. }
  677. case RTC_IOC_SET_TICK_MODE:
  678. {
  679. ptick = (RTC_TICK_T *) u32Arg0;
  680. if (g_bIsEnableTickInt == TRUE)
  681. {
  682. RTC_Ioctl(0, RTC_IOC_DISABLE_INT, RTC_TICK_INT, 0);
  683. g_bIsEnableTickInt = TRUE;
  684. }
  685. if (ptick->ucMode > RTC_TICK_1_128_SEC) /*Tick mode 0 to 7 */
  686. {
  687. return E_RTC_ERR_ENOTTY ;
  688. }
  689. RTC_WriteEnable(1);
  690. outp32(REG_RTC_TICK, ptick->ucMode);
  691. RTC_Check();
  692. /*---------------------------------------------------------------------------------------------*/
  693. /* Reset tick interrupt status if program enable tick interrupt before. */
  694. /*---------------------------------------------------------------------------------------------*/
  695. if (g_bIsEnableTickInt == TRUE)
  696. {
  697. RTC_Ioctl(0, RTC_IOC_ENABLE_INT, RTC_TICK_INT, 0);
  698. return E_RTC_SUCCESS;
  699. }
  700. break;
  701. }
  702. case RTC_IOC_GET_TICK:
  703. {
  704. break;
  705. }
  706. case RTC_IOC_RESTORE_TICK:
  707. {
  708. break;
  709. }
  710. case RTC_IOC_ENABLE_INT:
  711. {
  712. switch ((RTC_INT_SOURCE)u32Arg0)
  713. {
  714. case RTC_TICK_INT:
  715. {
  716. g_bIsEnableTickInt = TRUE;
  717. u32Tmp = inp32(REG_RTC_INTEN) | RTC_TICK_INT;
  718. break;
  719. }
  720. case RTC_ALARM_INT:
  721. {
  722. g_bIsEnableAlarmInt = TRUE;
  723. RTC_WriteEnable(1);
  724. u32Tmp = inp32(REG_RTC_PWRCTL) | RTC_PWRCTL_ALARM_EN_Msk;
  725. outp32(REG_RTC_PWRCTL, u32Tmp);
  726. outp32(REG_RTC_INTEN, inp32(REG_RTC_INTEN) | RTC_INTEN_ALMIEN_Msk);
  727. RTC_Check();
  728. u32Tmp = inp32(REG_RTC_INTEN) | RTC_ALARM_INT;
  729. break;
  730. }
  731. case RTC_RELATIVE_ALARM_INT:
  732. {
  733. g_bIsEnableAlarmInt = TRUE;
  734. RTC_WriteEnable(1);
  735. u32Tmp = inp32(REG_RTC_PWRCTL) | RTC_PWRCTL_REL_ALARM_EN_Msk;
  736. outp32(REG_RTC_PWRCTL, u32Tmp);
  737. RTC_Check();
  738. u32Tmp = inp32(REG_RTC_INTEN) | RTC_RELATIVE_ALARM_INT;
  739. break;
  740. }
  741. case RTC_PSWI_INT:
  742. {
  743. g_bIsEnableAlarmInt = TRUE;
  744. u32Tmp = inp32(REG_RTC_INTEN) | RTC_PSWI_INT;
  745. break;
  746. }
  747. default:
  748. {
  749. return E_RTC_ERR_ENOTTY;
  750. }
  751. }
  752. RTC_WriteEnable(1);
  753. outp32(REG_RTC_INTEN, u32Tmp);
  754. RTC_Check();
  755. break;
  756. }
  757. case RTC_IOC_DISABLE_INT:
  758. {
  759. switch ((RTC_INT_SOURCE)u32Arg0)
  760. {
  761. case RTC_TICK_INT:
  762. {
  763. g_bIsEnableTickInt = FALSE;
  764. RTC_WriteEnable(1);
  765. u32Tmp = inp32(REG_RTC_INTEN) & (~RTC_TICK_INT);
  766. outp32(REG_RTC_INTEN, u32Tmp);
  767. outp32(REG_RTC_INTSTS, RTC_TICK_INT);
  768. RTC_Check();
  769. break;
  770. }
  771. case RTC_ALARM_INT:
  772. {
  773. g_bIsEnableAlarmInt = FALSE;
  774. RTC_WriteEnable(1);
  775. u32Tmp = inp32(REG_RTC_INTEN) & (~RTC_ALARM_INT);
  776. outp32(REG_RTC_INTEN, u32Tmp);
  777. RTC_Check();
  778. RTC_WriteEnable(1);
  779. u32Tmp = inp32(REG_RTC_PWRCTL) & ~RTC_PWRCTL_ALARM_EN_Msk;
  780. outp32(REG_RTC_PWRCTL, u32Tmp);
  781. RTC_Check();
  782. outp32(REG_RTC_INTSTS, RTC_ALARM_INT);
  783. break;
  784. }
  785. case RTC_RELATIVE_ALARM_INT:
  786. {
  787. g_bIsEnableAlarmInt = FALSE;
  788. RTC_WriteEnable(1);
  789. u32Tmp = inp32(REG_RTC_INTEN) & (~RTC_RELATIVE_ALARM_INT);
  790. outp32(REG_RTC_INTEN, u32Tmp);
  791. RTC_Check();
  792. RTC_WriteEnable(1);
  793. u32Tmp = inp32(REG_RTC_PWRCTL) & ~RTC_PWRCTL_REL_ALARM_EN_Msk;
  794. outp32(REG_RTC_PWRCTL, u32Tmp);
  795. RTC_Check();
  796. outp32(REG_RTC_INTSTS, RTC_RELATIVE_ALARM_INT);
  797. break;
  798. }
  799. case RTC_PSWI_INT:
  800. {
  801. g_bIsEnableAlarmInt = FALSE;
  802. RTC_WriteEnable(1);
  803. u32Tmp = inp32(REG_RTC_INTEN) & (~RTC_PSWI_INT);
  804. outp32(REG_RTC_INTEN, u32Tmp);
  805. RTC_Check();
  806. outp32(REG_RTC_INTSTS, RTC_PSWI_INT);
  807. break;
  808. }
  809. case RTC_ALL_INT:
  810. {
  811. g_bIsEnableTickInt = FALSE;
  812. g_bIsEnableAlarmInt = FALSE;
  813. RTC_WriteEnable(1);
  814. outp32(REG_RTC_INTEN, 0);
  815. outp32(REG_RTC_INTSTS, RTC_ALL_INT);
  816. RTC_Check();
  817. break;
  818. }
  819. default:
  820. {
  821. return E_RTC_ERR_ENOTTY;
  822. }
  823. }
  824. break;
  825. }
  826. case RTC_IOC_SET_FREQUENCY:
  827. {
  828. i32Ret = RTC_DoFrequencyCompensation(u32Arg0) ;
  829. if (i32Ret != 0)
  830. {
  831. return E_RTC_ERR_ENOTTY;
  832. }
  833. break;
  834. }
  835. case RTC_IOC_SET_POWER_ON:
  836. {
  837. RTC_WriteEnable(1);
  838. u32Tmp = inp32(REG_RTC_PWRCTL) | 0x01;
  839. outp32(REG_RTC_PWRCTL, u32Tmp);
  840. RTC_Check();
  841. while ((inp32(REG_RTC_PWRCTL) & 0x01) != 0x1);
  842. break;
  843. }
  844. case RTC_IOC_SET_POWER_OFF:
  845. {
  846. RTC_WriteEnable(1);
  847. outp32(REG_RTC_PWRCTL, (inp32(REG_RTC_PWRCTL) & ~0x01) | 2);
  848. RTC_Check();
  849. while (1);
  850. //break;
  851. }
  852. case RTC_IOC_SET_POWER_OFF_PERIOD:
  853. {
  854. if (u32Arg0 < 4) u32Arg0 = 4;
  855. u32Arg0 = u32Arg0 - 4;
  856. RTC_WriteEnable(1);
  857. outp32(REG_RTC_PWRCTL, (inp32(REG_RTC_PWRCTL) & ~0xF000) | ((u32Arg0 & 0xF) << 12));
  858. RTC_Check();
  859. break;
  860. }
  861. case RTC_IOC_ENABLE_HW_POWEROFF:
  862. {
  863. RTC_WriteEnable(1);
  864. outp32(REG_RTC_PWRCTL, (inp32(REG_RTC_PWRCTL) | 0x04));
  865. RTC_Check();
  866. break;
  867. }
  868. case RTC_IOC_DISABLE_HW_POWEROFF:
  869. {
  870. RTC_WriteEnable(1);
  871. outp32(REG_RTC_PWRCTL, (inp32(REG_RTC_PWRCTL) & ~0x04));
  872. RTC_Check();
  873. break;
  874. }
  875. case RTC_IOC_SET_PSWI_CALLBACK:
  876. {
  877. RTC_Ioctl(0, RTC_IOC_ENABLE_INT, RTC_PSWI_INT, 0);
  878. break;
  879. }
  880. case RTC_IOC_GET_POWERKEY_STATUS:
  881. {
  882. RTC_WriteEnable(1);
  883. if (inp32(REG_RTC_PWRCTL) & 0x80)
  884. *(PUINT32)u32Arg0 = 1;
  885. else
  886. *(PUINT32)u32Arg0 = 0;
  887. break;
  888. }
  889. case RTC_IOC_SET_RELEATIVE_ALARM:
  890. {
  891. g_bIsEnableAlarmInt = TRUE;
  892. RTC_WriteEnable(1);
  893. outp32(REG_RTC_PWRCTL, (inp32(REG_RTC_PWRCTL) & ~0xFFF0010));
  894. RTC_Check();
  895. RTC_WriteEnable(1);
  896. u32Tmp = (inp32(REG_RTC_PWRCTL) & ~0xFFF0000) | ((u32Arg0 & 0xFFF) << 16) | RTC_PWRCTL_REL_ALARM_EN_Msk;
  897. outp32(REG_RTC_PWRCTL, u32Tmp);
  898. RTC_Check();
  899. g_bIsEnableAlarmInt = TRUE;
  900. RTC_WriteEnable(1);
  901. u32Tmp = inp32(REG_RTC_INTEN) | RTC_RELATIVE_ALARM_INT;
  902. outp32(REG_RTC_INTEN, u32Tmp);
  903. RTC_Check();
  904. break;
  905. }
  906. default:
  907. {
  908. return E_RTC_ERR_ENOTTY;
  909. }
  910. }
  911. return E_RTC_SUCCESS;
  912. }
  913. /**
  914. * @brief Disable AIC channel of RTC and both tick and alarm interrupt.
  915. *
  916. * @param[in] None
  917. *
  918. * @retval E_RTC_SUCCESS Success
  919. *
  920. */
  921. UINT32 RTC_Close(void)
  922. {
  923. g_bIsEnableTickInt = FALSE;
  924. sysDisableInterrupt(RTC_IRQn);
  925. RTC_Ioctl(0, RTC_IOC_DISABLE_INT, RTC_ALL_INT, 0);
  926. return E_RTC_SUCCESS;
  927. }
  928. /**
  929. * @brief Enable RTC clock.
  930. *
  931. * @param[in] bEnable 1: Enable \n
  932. * 2: Disable
  933. *
  934. * @return None
  935. *
  936. */
  937. void RTC_EnableClock(BOOL bEnable)
  938. {
  939. if (bEnable)
  940. outp32(REG_CLK_PCLKEN0, inp32(REG_CLK_PCLKEN0) | (1 << 2));
  941. else
  942. outp32(REG_CLK_PCLKEN0, inp32(REG_CLK_PCLKEN0) & ~(1 << 2));
  943. }
  944. /*@}*/ /* end of group N9H30_RTC_EXPORTED_FUNCTIONS */
  945. /*@}*/ /* end of group N9H30_RTC_Driver */
  946. /*@}*/ /* end of group N9H30_Device_Driver */
  947. /*** (C) COPYRIGHT 2018 Nuvoton Technology Corp. ***/