mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
255 lines
7.4 KiB
C
255 lines
7.4 KiB
C
|
/******************************************************************************
|
||
|
Copyright 2008-2010, Freie Universitaet Berlin (FUB). All rights reserved.
|
||
|
|
||
|
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 FeuerWare.
|
||
|
|
||
|
This program is free software: you can redistribute it and/or modify it under
|
||
|
the terms of the GNU General Public License as published by the Free Software
|
||
|
Foundation, either version 3 of the License, or (at your option) any later
|
||
|
version.
|
||
|
|
||
|
FeuerWare is distributed in the hope that it will be useful, but WITHOUT
|
||
|
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
||
|
FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
||
|
|
||
|
You should have received a copy of the GNU General Public License along with
|
||
|
this program. If not, see http://www.gnu.org/licenses/ .
|
||
|
--------------------------------------------------------------------------------
|
||
|
For further information and questions please use the web site
|
||
|
http://scatterweb.mi.fu-berlin.de
|
||
|
and the mailinglist (subscription via web site)
|
||
|
scatterweb@lists.spline.inf.fu-berlin.de
|
||
|
*******************************************************************************/
|
||
|
|
||
|
/**
|
||
|
* @file
|
||
|
* @ingroup lpc2387_rtc
|
||
|
* @brief LPC2387 Real-Time-Clock
|
||
|
*
|
||
|
* @author Freie Universität Berlin, Computer Systems & Telematics, FeuerWhere project
|
||
|
* @author Michael Baar <michael.baar@fu-berlin.de>
|
||
|
* @version $Revision: 2005 $
|
||
|
*
|
||
|
* @note $Id: lpc2387-rtc.c 2005 2010-03-17 10:52:03Z baar $
|
||
|
*/
|
||
|
|
||
|
#include <sys/time.h>
|
||
|
#include <stdint.h>
|
||
|
#include <string.h>
|
||
|
|
||
|
// cpu
|
||
|
#include "VIC.h"
|
||
|
#include "lpc2387.h"
|
||
|
#include "lpc2387-rtc.h"
|
||
|
#include "lpm.h"
|
||
|
#include "clock.h"
|
||
|
|
||
|
#define PREINT_RTC 0x000001C8 /* Prescaler value, integer portion, PCLK = 15Mhz */
|
||
|
#define PREFRAC_RTC 0x000061C0 /* Prescaler value, fraction portion, PCLK = 15Mhz */
|
||
|
|
||
|
#define DEBUG 0
|
||
|
#if DEBUG
|
||
|
#include <stdio.h>
|
||
|
#define PRINTF(fmt, args...) printf("rtc: " fmt "\n", ##args)
|
||
|
#else
|
||
|
#define PRINTF(fmt, args...)
|
||
|
#endif
|
||
|
|
||
|
extern void _clock_alarm(void);
|
||
|
|
||
|
/**
|
||
|
* @brief epoch time in hour granularity
|
||
|
*/
|
||
|
static volatile time_t epoch;
|
||
|
|
||
|
/*---------------------------------------------------------------------------*/
|
||
|
/**
|
||
|
* @brief Sets the current time in broken down format directly from to RTC
|
||
|
* @param[in] localt Pointer to structure with time to set
|
||
|
*/
|
||
|
static void
|
||
|
rtc_set_localtime(struct tm* localt)
|
||
|
{
|
||
|
if( localt == NULL )
|
||
|
return;
|
||
|
|
||
|
/* set clock */
|
||
|
RTC_SEC = localt->tm_sec;
|
||
|
RTC_MIN = localt->tm_min;
|
||
|
RTC_HOUR = localt->tm_hour;
|
||
|
RTC_DOM = localt->tm_mday;
|
||
|
RTC_DOW = localt->tm_wday;
|
||
|
RTC_DOY = localt->tm_yday;
|
||
|
RTC_MONTH = localt->tm_mon + 1;
|
||
|
RTC_YEAR = localt->tm_year;
|
||
|
}
|
||
|
/*---------------------------------------------------------------------------*/
|
||
|
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;
|
||
|
}
|
||
|
/*---------------------------------------------------------------------------*/
|
||
|
/// set clock to start of unix epoch
|
||
|
void _rtc_reset(void)
|
||
|
{
|
||
|
rtc_set(0);
|
||
|
epoch = 0;
|
||
|
}
|
||
|
/*---------------------------------------------------------------------------*/
|
||
|
void
|
||
|
_rtc_set_alarm(struct tm* localt, enum rtc_alarm_mask mask)
|
||
|
{
|
||
|
if( localt != NULL ) {
|
||
|
RTC_ALSEC = localt->tm_sec;
|
||
|
RTC_ALMIN = localt->tm_min;
|
||
|
RTC_ALHOUR = localt->tm_hour;
|
||
|
RTC_ALDOM = localt->tm_mday;
|
||
|
RTC_ALDOW = localt->tm_wday;
|
||
|
RTC_ALDOY = localt->tm_yday;
|
||
|
RTC_ALMON = localt->tm_mon + 1;
|
||
|
RTC_ALYEAR = localt->tm_year;
|
||
|
RTC_AMR = ~mask; // set wich alarm fields to check
|
||
|
PRINTF("alarm set %2lu.%2lu.%4lu %2lu:%2lu:%2lu",
|
||
|
RTC_ALDOM, RTC_ALMON, RTC_ALYEAR, RTC_ALHOUR, RTC_ALMIN, RTC_ALSEC);
|
||
|
} else {
|
||
|
RTC_AMR = 0xff;
|
||
|
}
|
||
|
}
|
||
|
/*---------------------------------------------------------------------------*/
|
||
|
enum rtc_alarm_mask
|
||
|
_rtc_get_alarm(struct tm* localt)
|
||
|
{
|
||
|
if( localt != NULL ) {
|
||
|
localt->tm_sec = RTC_ALSEC;
|
||
|
localt->tm_min = RTC_ALMIN;
|
||
|
localt->tm_hour = RTC_ALHOUR;
|
||
|
localt->tm_mday = RTC_ALDOM;
|
||
|
localt->tm_wday = RTC_ALDOW;
|
||
|
localt->tm_yday = RTC_ALDOY;
|
||
|
localt->tm_mon = RTC_ALMON - 1;
|
||
|
localt->tm_year = RTC_ALYEAR;
|
||
|
localt->tm_isdst = -1; // not available
|
||
|
}
|
||
|
return (~RTC_AMR) & 0xff; // return which alarm fields are checked
|
||
|
}
|
||
|
/*---------------------------------------------------------------------------*/
|
||
|
void RTC_IRQHandler (void) __attribute__ ((interrupt("IRQ")));
|
||
|
void RTC_IRQHandler (void)
|
||
|
{
|
||
|
lpm_begin_awake();
|
||
|
if( RTC_ILR & ILR_RTSSF ) {
|
||
|
// sub second interrupt (does not need flag-clearing)
|
||
|
|
||
|
} else if( RTC_ILR & ILR_RTCCIF ) {
|
||
|
// counter increase interrupt
|
||
|
RTC_ILR |= ILR_RTCCIF;
|
||
|
epoch += 60 * 60; // add 1 hour
|
||
|
|
||
|
} else if( RTC_ILR & ILR_RTCALF ) {
|
||
|
RTC_ILR |= ILR_RTCALF;
|
||
|
RTC_AMR = 0xff; // disable alarm irq
|
||
|
PRINTF("alarm");
|
||
|
lpm_end_awake();
|
||
|
_clock_alarm();
|
||
|
}
|
||
|
|
||
|
VICVectAddr = 0; // Acknowledge Interrupt
|
||
|
}
|
||
|
|
||
|
/*---------------------------------------------------------------------------*/
|
||
|
void rtc_enable(void)
|
||
|
{
|
||
|
RTC_ILR = (ILR_RTSSF | ILR_RTCCIF | ILR_RTCALF); // clear interrupt flags
|
||
|
RTC_CCR |= CCR_CLKEN; // enable clock
|
||
|
install_irq(RTC_INT, &RTC_IRQHandler, IRQP_RTC); // install interrupt handler
|
||
|
|
||
|
time_t now = rtc_time(NULL);
|
||
|
epoch = now - (now % 3600);
|
||
|
}
|
||
|
/*---------------------------------------------------------------------------*/
|
||
|
void _rtc_init(void)
|
||
|
{
|
||
|
PCONP |= BIT9;
|
||
|
RTC_AMR = 0xff; // disable alarm irq
|
||
|
RTC_CIIR = IMHOUR; // enable increase irq
|
||
|
RTC_CISS = 0; // disable subsecond irq
|
||
|
|
||
|
INTWAKE |= BIT15; // rtc irq wakes up mcu from power down
|
||
|
|
||
|
RTC_CCR = CCR_CLKSRC; // Clock from external 32 kHz Osc.
|
||
|
|
||
|
/* initialize clock with valid unix compatible values
|
||
|
* If RTC_YEAR contains an value larger unix time_t we must reset. */
|
||
|
if( RTC_YEAR > 2037 ) {
|
||
|
_rtc_reset();
|
||
|
}
|
||
|
|
||
|
PRINTF("%2lu.%2lu.%4lu %2lu:%2lu:%2lu epoch %lu",
|
||
|
RTC_DOM, RTC_MONTH, RTC_YEAR, RTC_HOUR, RTC_MIN, RTC_SEC,
|
||
|
epoch);
|
||
|
}
|
||
|
/*---------------------------------------------------------------------------*/
|
||
|
time_t rtc_time(struct timeval* time)
|
||
|
{
|
||
|
uint32_t sec;
|
||
|
uint32_t usec;
|
||
|
uint32_t min;
|
||
|
|
||
|
usec = (RTC_CTC >> 1);
|
||
|
sec = RTC_SEC;
|
||
|
min = RTC_MIN;
|
||
|
while (usec != (RTC_CTC>>1) ) {
|
||
|
usec = (RTC_CTC >> 1);
|
||
|
sec = RTC_SEC;
|
||
|
min = RTC_MIN;
|
||
|
}
|
||
|
|
||
|
sec += min * 60; // add number of minutes
|
||
|
sec += epoch; // add precalculated epoch in hour granularity
|
||
|
|
||
|
if( time != NULL ) {
|
||
|
usec = usec * 15625;
|
||
|
usec >>= 9;
|
||
|
time->tv_sec = sec;
|
||
|
time->tv_usec = usec;
|
||
|
}
|
||
|
|
||
|
return sec;
|
||
|
}
|
||
|
/*---------------------------------------------------------------------------*/
|
||
|
void rtc_disable(void)
|
||
|
{
|
||
|
RTC_CCR &= ~CCR_CLKEN; // disable clock
|
||
|
install_irq(RTC_INT, NULL, 0);
|
||
|
RTC_ILR = 0;
|
||
|
}
|
||
|
/*---------------------------------------------------------------------------*/
|
||
|
void
|
||
|
rtc_get_localtime(struct tm* localt)
|
||
|
{
|
||
|
if( localt != NULL ) {
|
||
|
localt->tm_sec = RTC_SEC;
|
||
|
localt->tm_min = RTC_MIN;
|
||
|
localt->tm_hour = RTC_HOUR;
|
||
|
localt->tm_mday = RTC_DOM;
|
||
|
localt->tm_wday = RTC_DOW;
|
||
|
localt->tm_yday = RTC_DOY;
|
||
|
localt->tm_mon = RTC_MONTH - 1;
|
||
|
localt->tm_year = RTC_YEAR;
|
||
|
localt->tm_isdst = -1; // not available
|
||
|
}
|
||
|
}
|
||
|
/*---------------------------------------------------------------------------*/
|
||
|
void _gettimeofday_r(struct _reent *r, struct timeval *ptimeval, struct timezone *ptimezone)
|
||
|
{
|
||
|
r->_errno = 0;
|
||
|
if( ptimeval != NULL ) {
|
||
|
rtc_time(ptimeval);
|
||
|
}
|
||
|
}
|