| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609 | 
							- /*
 
-  * Copyright (c) 2006-2018, RT-Thread Development Team
 
-  *
 
-  * SPDX-License-Identifier: Apache-2.0
 
-  *
 
-  * Change Logs:
 
-  * Date           Author            Notes
 
-  * 2012-10-27     heyuanjie87       first version.
 
-  * 2013-05-17     aozima            initial alarm event & mutex in system init.
 
-  */
 
- #include <rtthread.h>
 
- #include <rtdevice.h>
 
- #define RT_RTC_YEARS_MAX         137
 
- #define RT_ALARM_DELAY             2
 
- #define RT_ALARM_STATE_INITED   0x02
 
- #define RT_ALARM_STATE_START    0x01
 
- #define RT_ALARM_STATE_STOP     0x00
 
- #if (defined(RT_USING_RTC) && defined(RT_USING_ALARM))
 
- static struct rt_alarm_container _container;
 
- rt_inline rt_uint32_t alarm_mkdaysec(struct tm *time)
 
- {
 
-     rt_uint32_t sec;
 
-     sec = time->tm_sec;
 
-     sec += time->tm_min * 60;
 
-     sec += time->tm_hour * 3600;
 
-     return (sec);
 
- }
 
- static rt_err_t alarm_set(struct rt_alarm *alarm)
 
- {
 
-     rt_device_t device;
 
-     struct rt_rtc_wkalarm wkalarm;
 
-     rt_err_t ret;
 
-     device = rt_device_find("rtc");
 
-     if (device == RT_NULL)
 
-     {
 
-         return (RT_ERROR);
 
-     }
 
-     if (alarm->flag & RT_ALARM_STATE_START)
 
-         wkalarm.enable = RT_TRUE;
 
-     else
 
-         wkalarm.enable = RT_FALSE;
 
-     wkalarm.tm_sec = alarm->wktime.tm_sec;
 
-     wkalarm.tm_min = alarm->wktime.tm_min;
 
-     wkalarm.tm_hour = alarm->wktime.tm_hour;
 
-     ret = rt_device_control(device, RT_DEVICE_CTRL_RTC_SET_ALARM, &wkalarm);
 
-     if ((ret == RT_EOK) && wkalarm.enable)
 
-     {
 
-         ret = rt_device_control(device, RT_DEVICE_CTRL_RTC_GET_ALARM, &wkalarm);
 
-         if (ret == RT_EOK)
 
-         {
 
-             /*
 
-               some RTC device like RX8025,it's alarms precision is 1 minute.
 
-               in this case,low level RTC driver should set wkalarm->tm_sec to 0.
 
-             */
 
-             alarm->wktime.tm_sec = wkalarm.tm_sec;
 
-             alarm->wktime.tm_min = wkalarm.tm_min;
 
-             alarm->wktime.tm_hour = wkalarm.tm_hour;
 
-         }
 
-     }
 
-     return (ret);
 
- }
 
- static void alarm_wakeup(struct rt_alarm *alarm, struct tm *now)
 
- {
 
-     rt_uint32_t sec_alarm, sec_now;
 
-     rt_bool_t wakeup = RT_FALSE;
 
-     time_t timestamp;
 
-     sec_alarm = alarm_mkdaysec(&alarm->wktime);
 
-     sec_now = alarm_mkdaysec(now);
 
-     if (alarm->flag & RT_ALARM_STATE_START)
 
-     {
 
-         switch (alarm->flag & 0xFF00)
 
-         {
 
-         case RT_ALARM_ONESHOT:
 
-         {
 
-             sec_alarm = mktime(&alarm->wktime);
 
-             sec_now = mktime(now);
 
-             if (((sec_now - sec_alarm) <= RT_ALARM_DELAY) && (sec_now >= sec_alarm))
 
-             {
 
-                 /* stop alarm */
 
-                 alarm->flag &= ~RT_ALARM_STATE_START;
 
-                 alarm_set(alarm);
 
-                 wakeup = RT_TRUE;
 
-             }
 
-         }
 
-         break;
 
-         case RT_ALARM_DAILY:
 
-         {
 
-             if (((sec_now - sec_alarm) <= RT_ALARM_DELAY) && (sec_now >= sec_alarm))
 
-                 wakeup = RT_TRUE;
 
-         }
 
-         break;
 
-         case RT_ALARM_WEEKLY:
 
-         {
 
-             /* alarm at wday */
 
-             sec_alarm += alarm->wktime.tm_wday * 24 * 3600;
 
-             sec_now += now->tm_wday * 24 * 3600;
 
-             if (((sec_now - sec_alarm) <= RT_ALARM_DELAY) && (sec_now >= sec_alarm))
 
-                 wakeup = RT_TRUE;
 
-         }
 
-         break;
 
-         case RT_ALARM_MONTHLY:
 
-         {
 
-             /* monthly someday generate alarm signals */
 
-             if (alarm->wktime.tm_mday == now->tm_mday)
 
-             {
 
-                 if ((sec_now - sec_alarm) <= RT_ALARM_DELAY)
 
-                     wakeup = RT_TRUE;
 
-             }
 
-         }
 
-         break;
 
-         case RT_ALARM_YAERLY:
 
-         {
 
-             if ((alarm->wktime.tm_mday == now->tm_mday) && \
 
-                     (alarm->wktime.tm_mon == now->tm_mon))
 
-             {
 
-                 if ((sec_now - sec_alarm) <= RT_ALARM_DELAY)
 
-                     wakeup = RT_TRUE;
 
-             }
 
-         }
 
-         break;
 
-         }
 
-         if ((wakeup == RT_TRUE) && (alarm->callback != RT_NULL))
 
-         {
 
-             timestamp = time(RT_NULL);
 
-             alarm->callback(alarm, timestamp);
 
-         }
 
-     }
 
- }
 
- static void alarm_update(rt_uint32_t event)
 
- {
 
-     struct rt_alarm *alm_prev = RT_NULL, *alm_next = RT_NULL;
 
-     struct rt_alarm *alarm;
 
-     rt_int32_t sec_now, sec_alarm, sec_tmp;
 
-     rt_int32_t sec_next = 24 * 3600, sec_prev = 0;
 
-     time_t timestamp;
 
-     struct tm now;
 
-     rt_list_t *next;
 
-     rt_mutex_take(&_container.mutex, RT_WAITING_FOREVER);
 
-     if (!rt_list_isempty(&_container.head))
 
-     {
 
-         /* get time of now */
 
-         timestamp = time(RT_NULL);
 
-         localtime_r(×tamp, &now);
 
-         for (next = _container.head.next; next != &_container.head; next = next->next)
 
-         {
 
-             alarm = rt_list_entry(next, struct rt_alarm, list);
 
-             /* check the overtime alarm */
 
-             alarm_wakeup(alarm, &now);
 
-         }
 
-         timestamp = time(RT_NULL);
 
-         localtime_r(×tamp, &now);
 
-         sec_now = alarm_mkdaysec(&now);
 
-         for (next = _container.head.next; next != &_container.head; next = next->next)
 
-         {
 
-             alarm = rt_list_entry(next, struct rt_alarm, list);
 
-             /* calculate seconds from 00:00:00 */
 
-             sec_alarm = alarm_mkdaysec(&alarm->wktime);
 
-             if ((alarm->flag & RT_ALARM_STATE_START) && (alarm != _container.current))
 
-             {
 
-                 sec_tmp = sec_alarm - sec_now;
 
-                 if (sec_tmp > 0)
 
-                 {
 
-                     /* find alarm after now(now to 23:59:59) and the most recent */
 
-                     if (sec_tmp < sec_next)
 
-                     {
 
-                         sec_next = sec_tmp;
 
-                         alm_next = alarm;
 
-                     }
 
-                 }
 
-                 else
 
-                 {
 
-                     /* find alarm before now(00:00:00 to now) and furthest from now */
 
-                     if (sec_tmp < sec_prev)
 
-                     {
 
-                         sec_prev = sec_tmp;
 
-                         alm_prev = alarm;
 
-                     }
 
-                 }
 
-             }
 
-         }
 
-         /* enable the alarm after now first */
 
-         if (sec_next < 24 * 3600)
 
-         {
 
-             if (alarm_set(alm_next) == RT_EOK)
 
-                 _container.current = alm_next;
 
-         }
 
-         else if (sec_prev < 0)
 
-         {
 
-             /* enable the alarm before now */
 
-             if (alarm_set(alm_prev) == RT_EOK)
 
-                 _container.current = alm_prev;
 
-         }
 
-     }
 
-     rt_mutex_release(&_container.mutex);
 
- }
 
- static rt_uint32_t days_of_year_month(int tm_year, int tm_mon)
 
- {
 
-     rt_uint32_t ret, year;
 
-     year = tm_year + 1900;
 
-     if (tm_mon == 1)
 
-     {
 
-         ret = 28 + ((!(year % 4) && (year % 100)) || !(year % 400));
 
-     }
 
-     else if (((tm_mon <= 6) && (tm_mon % 2 == 0)) || ((tm_mon > 6) && (tm_mon % 2 == 1)))
 
-     {
 
-         ret = 31;
 
-     }
 
-     else
 
-     {
 
-         ret = 30;
 
-     }
 
-     return (ret);
 
- }
 
- static rt_bool_t is_valid_date(struct tm *date)
 
- {
 
-     if ((date->tm_year < 0) || (date->tm_year > RT_RTC_YEARS_MAX))
 
-     {
 
-         return (RT_FALSE);
 
-     }
 
-     if ((date->tm_mon < 0) || (date->tm_mon > 11))
 
-     {
 
-         return (RT_FALSE);
 
-     }
 
-     if ((date->tm_mday < 1) || \
 
-             (date->tm_mday > days_of_year_month(date->tm_year, date->tm_mon)))
 
-     {
 
-         return (RT_FALSE);
 
-     }
 
-     return (RT_TRUE);
 
- }
 
- static rt_err_t alarm_setup(rt_alarm_t alarm, struct tm *wktime)
 
- {
 
-     rt_err_t ret = RT_ERROR;
 
-     time_t timestamp;
 
-     struct tm *setup, now;
 
-     setup = &alarm->wktime;
 
-     *setup = *wktime;
 
-     timestamp = time(RT_NULL);
 
-     localtime_r(×tamp, &now);
 
-     /* if these are a "don't care" value,we set them to now*/
 
-     if ((setup->tm_sec > 59) || (setup->tm_sec < 0))
 
-         setup->tm_sec = now.tm_sec;
 
-     if ((setup->tm_min > 59) || (setup->tm_min < 0))
 
-         setup->tm_min = now.tm_min;
 
-     if ((setup->tm_hour > 23) || (setup->tm_hour < 0))
 
-         setup->tm_hour = now.tm_hour;
 
-     switch (alarm->flag & 0xFF00)
 
-     {
 
-     case RT_ALARM_DAILY:
 
-     {
 
-         /* do nothing but needed */
 
-     }
 
-     break;
 
-     case RT_ALARM_ONESHOT:
 
-     {
 
-         /* if these are "don't care" value we set them to now */
 
-         if (setup->tm_year == RT_ALARM_TM_NOW)
 
-             setup->tm_year = now.tm_year;
 
-         if (setup->tm_mon == RT_ALARM_TM_NOW)
 
-             setup->tm_mon = now.tm_mon;
 
-         if (setup->tm_mday == RT_ALARM_TM_NOW)
 
-             setup->tm_mday = now.tm_mday;
 
-         /* make sure the setup is valid */
 
-         if (!is_valid_date(setup))
 
-             goto _exit;
 
-     }
 
-     break;
 
-     case RT_ALARM_WEEKLY:
 
-     {
 
-         /* if tm_wday is a "don't care" value we set it to now */
 
-         if ((setup->tm_wday < 0) || (setup->tm_wday > 6))
 
-             setup->tm_wday = now.tm_wday;
 
-     }
 
-     break;
 
-     case RT_ALARM_MONTHLY:
 
-     {
 
-         /* if tm_mday is a "don't care" value we set it to now */
 
-         if ((setup->tm_mday < 1) || (setup->tm_mday > 31))
 
-             setup->tm_mday = now.tm_mday;
 
-     }
 
-     break;
 
-     case RT_ALARM_YAERLY:
 
-     {
 
-         /* if tm_mon is a "don't care" value we set it to now */
 
-         if ((setup->tm_mon < 0) || (setup->tm_mon > 11))
 
-             setup->tm_mon = now.tm_mon;
 
-         if (setup->tm_mon == 1)
 
-         {
 
-             /* tm_mon is February */
 
-             /* tm_mday should be 1~29.otherwise,it's a "don't care" value */
 
-             if ((setup->tm_mday < 1) || (setup->tm_mday > 29))
 
-                 setup->tm_mday = now.tm_mday;
 
-         }
 
-         else if (((setup->tm_mon <= 6) && (setup->tm_mon % 2 == 0)) || \
 
-                  ((setup->tm_mon > 6) && (setup->tm_mon % 2 == 1)))
 
-         {
 
-             /* Jan,Mar,May,Jul,Aug,Oct,Dec */
 
-             /* tm_mday should be 1~31.otherwise,it's a "don't care" value */
 
-             if ((setup->tm_mday < 1) || (setup->tm_mday > 31))
 
-                 setup->tm_mday = now.tm_mday;
 
-         }
 
-         else
 
-         {
 
-             /* tm_mday should be 1~30.otherwise,it's a "don't care" value */
 
-             if ((setup->tm_mday < 1) || (setup->tm_mday > 30))
 
-                 setup->tm_mday = now.tm_mday;
 
-         }
 
-     }
 
-     break;
 
-     default:
 
-     {
 
-         goto _exit;
 
-     }
 
-     }
 
-     if ((setup->tm_hour == 23) && (setup->tm_min == 59) && (setup->tm_sec == 59))
 
-     {
 
-         /*
 
-            for insurance purposes, we will generate an alarm
 
-            signal two seconds ahead of.
 
-         */
 
-         setup->tm_sec = 60 - RT_ALARM_DELAY;
 
-     }
 
-     /* set initialized state */
 
-     alarm->flag |= RT_ALARM_STATE_INITED;
 
-     ret = RT_EOK;
 
- _exit:
 
-     return (ret);
 
- }
 
- /** \brief send a rtc alarm event
 
-  *
 
-  * \param dev pointer to RTC device(currently unused,you can ignore it)
 
-  * \param event RTC event(currently unused)
 
-  * \return none
 
-  */
 
- void rt_alarm_update(rt_device_t dev, rt_uint32_t event)
 
- {
 
-     rt_event_send(&_container.event, 1);
 
- }
 
- /** \brief modify the alarm setup
 
-  *
 
-  * \param alarm pointer to alarm
 
-  * \param cmd control command
 
-  * \param arg argument
 
-  */
 
- rt_err_t rt_alarm_control(rt_alarm_t alarm, int cmd, void *arg)
 
- {
 
-     rt_err_t ret = RT_ERROR;
 
-     RT_ASSERT(alarm != RT_NULL);
 
-     rt_mutex_take(&_container.mutex, RT_WAITING_FOREVER);
 
-     switch (cmd)
 
-     {
 
-     case RT_ALARM_CTRL_MODIFY:
 
-     {
 
-         struct rt_alarm_setup *setup;
 
-         RT_ASSERT(arg != RT_NULL);
 
-         setup = arg;
 
-         rt_alarm_stop(alarm);
 
-         alarm->flag = setup->flag & 0xFF00;
 
-         alarm->wktime = setup->wktime;
 
-         ret = alarm_setup(alarm, &alarm->wktime);
 
-     }
 
-     break;
 
-     }
 
-     rt_mutex_release(&_container.mutex);
 
-     return (ret);
 
- }
 
- /** \brief start an alarm
 
-  *
 
-  * \param alarm pointer to alarm
 
-  * \return RT_EOK
 
-  */
 
- rt_err_t rt_alarm_start(rt_alarm_t alarm)
 
- {
 
-     rt_int32_t sec_now, sec_old, sec_new;
 
-     rt_err_t ret = RT_ERROR;
 
-     time_t timestamp;
 
-     struct tm now;
 
-     if (alarm == RT_NULL)
 
-         return (ret);
 
-     rt_mutex_take(&_container.mutex, RT_WAITING_FOREVER);
 
-     if (!(alarm->flag & RT_ALARM_STATE_INITED))
 
-     {
 
-         if (alarm_setup(alarm, &alarm->wktime) != RT_EOK)
 
-             goto _exit;
 
-     }
 
-     if ((alarm->flag & 0x01) == RT_ALARM_STATE_STOP)
 
-     {
 
-         timestamp = time(RT_NULL);
 
-         localtime_r(×tamp, &now);
 
-         alarm->flag |= RT_ALARM_STATE_START;
 
-         /* set alarm */
 
-         if (_container.current == RT_NULL)
 
-         {
 
-             ret = alarm_set(alarm);
 
-         }
 
-         else
 
-         {
 
-             sec_now = alarm_mkdaysec(&now);
 
-             sec_old = alarm_mkdaysec(&_container.current->wktime);
 
-             sec_new = alarm_mkdaysec(&alarm->wktime);
 
-             if ((sec_new < sec_old) && (sec_new > sec_now))
 
-             {
 
-                 ret = alarm_set(alarm);
 
-             }
 
-             else if ((sec_new > sec_now) && (sec_old < sec_now))
 
-             {
 
-                 ret = alarm_set(alarm);
 
-             }
 
-             else if ((sec_new < sec_old) && (sec_old < sec_now))
 
-             {
 
-                 ret = alarm_set(alarm);
 
-             }
 
-             else
 
-             {
 
-                 ret = RT_EOK;
 
-                 goto _exit;
 
-             }
 
-         }
 
-         if (ret == RT_EOK)
 
-         {
 
-             _container.current = alarm;
 
-         }
 
-     }
 
- _exit:
 
-     rt_mutex_release(&_container.mutex);
 
-     return (ret);
 
- }
 
- /** \brief stop an alarm
 
-  *
 
-  * \param alarm pointer to alarm
 
-  * \return RT_EOK
 
-  */
 
- rt_err_t rt_alarm_stop(rt_alarm_t alarm)
 
- {
 
-     rt_err_t ret = RT_ERROR;
 
-     if (alarm == RT_NULL)
 
-         return (ret);
 
-     rt_mutex_take(&_container.mutex, RT_WAITING_FOREVER);
 
-     if (!(alarm->flag & RT_ALARM_STATE_START))
 
-         goto _exit;
 
-     /* stop alarm */
 
-     alarm->flag &= ~RT_ALARM_STATE_START;
 
-     if (_container.current == alarm)
 
-     {
 
-         ret = alarm_set(alarm);
 
-         _container.current = RT_NULL;
 
-     }
 
-     if (ret == RT_EOK)
 
-         alarm_update(0);
 
- _exit:
 
-     rt_mutex_release(&_container.mutex);
 
-     return (ret);
 
- }
 
- /** \brief delete an alarm
 
-  *
 
-  * \param alarm pointer to alarm
 
-  * \return RT_EOK
 
-  */
 
- rt_err_t rt_alarm_delete(rt_alarm_t alarm)
 
- {
 
-     rt_err_t ret = RT_ERROR;
 
-     if (alarm == RT_NULL)
 
-         return (ret);
 
-     rt_mutex_take(&_container.mutex, RT_WAITING_FOREVER);
 
-     /* stop the alarm */
 
-     alarm->flag &= ~RT_ALARM_STATE_START;
 
-     if (_container.current == alarm)
 
-     {
 
-         ret = alarm_set(alarm);
 
-         _container.current = RT_NULL;
 
-         /* set new alarm if necessary */
 
-         alarm_update(0);
 
-     }
 
-     rt_list_remove(&alarm->list);
 
-     rt_free(alarm);
 
-     rt_mutex_release(&_container.mutex);
 
-     return (ret);
 
- }
 
- /** \brief create an alarm
 
-  *
 
-  * \param flag set alarm mode e.g: RT_ALARM_DAILY
 
-  * \param setup pointer to setup infomation
 
-  */
 
- rt_alarm_t rt_alarm_create(rt_alarm_callback_t callback, struct rt_alarm_setup *setup)
 
- {
 
-     struct rt_alarm *alarm;
 
-     if (setup == RT_NULL)
 
-         return (RT_NULL);
 
-     alarm = rt_malloc(sizeof(struct rt_alarm));
 
-     if (alarm == RT_NULL)
 
-         return (RT_NULL);
 
-     rt_list_init(&alarm->list);
 
-     alarm->wktime = setup->wktime;
 
-     alarm->flag = setup->flag & 0xFF00;
 
-     alarm->callback = callback;
 
-     rt_mutex_take(&_container.mutex, RT_WAITING_FOREVER);
 
-     rt_list_insert_after(&_container.head, &alarm->list);
 
-     rt_mutex_release(&_container.mutex);
 
-     return (alarm);
 
- }
 
- /** \brief rtc alarm service thread entry
 
-  *
 
-  */
 
- static void rt_alarmsvc_thread_init(void *param)
 
- {
 
-     rt_uint32_t recv;
 
-     _container.current = RT_NULL;
 
-     while (1)
 
-     {
 
-         if (rt_event_recv(&_container.event, 0xFFFF,
 
-                           RT_EVENT_FLAG_OR | RT_EVENT_FLAG_CLEAR,
 
-                           RT_WAITING_FOREVER, &recv) == RT_EOK)
 
-         {
 
-             alarm_update(recv);
 
-         }
 
-     }
 
- }
 
- /** \brief initialize alarm service system
 
-  *
 
-  * \param none
 
-  * \return none
 
-  */
 
- void rt_alarm_system_init(void)
 
- {
 
-     rt_thread_t tid;
 
-     rt_list_init(&_container.head);
 
-     rt_event_init(&_container.event, "alarmsvc", RT_IPC_FLAG_FIFO);
 
-     rt_mutex_init(&_container.mutex, "alarmsvc", RT_IPC_FLAG_FIFO);
 
-     tid = rt_thread_create("alarmsvc",
 
-                            rt_alarmsvc_thread_init, RT_NULL,
 
-                            512, 8, 1);
 
-     if (tid != RT_NULL)
 
-         rt_thread_startup(tid);
 
- }
 
- #endif
 
 
  |