/****************************************************************************** Copyright 2010, Freie Universitaet Berlin (FUB). Copyright 2013, INRIA. These sources were developed at the Freie Universitaet Berlin, Computer Systems and Telematics group (http://cst.mi.fu-berlin.de). ------------------------------------------------------------------------------- This file is part of RIOT. This file is subject to the terms and conditions of the GNU Lesser General Public License. See the file LICENSE in the top level directory for more details. *******************************************************************************/ /** * @ingroup rtc * @file cc430-rtc.c * @brief CC430 real time clock implementation * @author Oliver Hahm */ #include #include #include #include #include #include //static volatile time_t epoch; static struct tm time_to_set; static int set_time = 0; int rtc_second_pid = 0; /*---------------------------------------------------------------------------*/ void rtc_init(void) { /* Set to calendar mode */ RTCCTL1 |= RTCMODE_H; } /*---------------------------------------------------------------------------*/ void rtc_enable(void) { /* Set RTC operational */ RTCCTL1 &= ~RTCHOLD_H; } /*---------------------------------------------------------------------------*/ void rtc_disable(void) { /* Stop RTC */ RTCCTL1 |= RTCHOLD_H; } /*---------------------------------------------------------------------------*/ void rtc_set_localtime(struct tm *localt) { if (localt == NULL) { return; } /* copy time to be set */ memcpy(&time_to_set, localt, sizeof(struct tm)); set_time = 1; } /*--------------------------------------------------------------------------- void rtc_set(time_t time) { struct tm* localt; localt = localtime(&time); // convert seconds to broken-down time rtc_set_localtime(localt); epoch = time - localt->tm_sec - localt->tm_min * 60; } */ /*--------------------------------------------------------------------------- time_t rtc_time(void) { time_t sec; struct tm t; rtc_get_localtime(&t); sec = mktime(&t); return sec; } */ /*---------------------------------------------------------------------------*/ void rtc_get_localtime(struct tm *localt) { uint8_t success = 0; uint8_t i; uint16_t tmpyear; if (localt == NULL) { return; } while (!success) { for (i = 0; i < 8; i++) { /* try again when RTC is in transition */ if (!(RTCCTL1 & RTCRDY_H)) { break; } switch(i) { case 0: localt->tm_sec = RTCSEC; break; case 1: localt->tm_min = RTCMIN; break; case 2: localt->tm_hour = RTCHOUR; break; case 3: localt->tm_mday = RTCDAY; break; case 4: localt->tm_wday = RTCDOW; break; case 5: localt->tm_mon = RTCMON - 1; break; case 6: tmpyear = RTCYEARL; tmpyear |= (RTCYEARH << 0x08); localt->tm_year = tmpyear - 1900; break; default: success = 1; break; } } } } /*---------------------------------------------------------------------------*/ void rtc_set_alarm(struct tm *localt, rtc_alarm_mask_t mask) { if (mask & RTC_ALARM_MIN) { RTCAMIN = localt->tm_min; RTCAMIN |= BIT7; } if (mask & RTC_ALARM_HOUR) { RTCAHOUR = localt->tm_hour; RTCAHOUR |= BIT7; } if (mask & RTC_ALARM_DOW) { RTCADOW = localt->tm_wday; RTCADOW |= BIT7; } if (mask & RTC_ALARM_DOM) { RTCADAY = localt->tm_mday; RTCADAY |= BIT7; } RTCCTL0 |= RTCAIE; } /*---------------------------------------------------------------------------*/ void rtc_remove_alarm(void) { /* reset all AE bits */ RTCAHOUR &= ~BIT7; RTCAMIN &= ~BIT7; RTCADAY &= ~BIT7; RTCADOW &= ~BIT7; /* reset alarm interrupt enable */ RTCCTL0 &= ~RTCAIE; } /*---------------------------------------------------------------------------*/ interrupt(RTC_VECTOR) __attribute__((naked)) rtc_isr(void) { __enter_isr(); /* RTC is save to write for up to one second now */ if (RTCIV == RTC_RTCRDYIFG) { /* disable interrupt */ //RTCCTL0 &= ~RTCRDYIE; if (set_time) { set_time = 0; /* set previous set time and reset it */ RTCSEC = time_to_set.tm_sec; RTCMIN = time_to_set.tm_min; RTCHOUR = time_to_set.tm_hour; RTCDAY = time_to_set.tm_mday; RTCDOW = time_to_set.tm_wday; RTCMON = time_to_set.tm_mon + 1; RTCYEARL = (time_to_set.tm_year + 1900) & 0xFF; RTCYEARH = (time_to_set.tm_year + 1900) >> 0x08; } if (rtc_second_pid) { msg_t m; m.type = RTC_SECOND; msg_send_int(&m, rtc_second_pid); } } /* RTC alarm */ else if (RTCIV == RTC_RTCAIFG) { } __exit_isr(); }