rockbox/firmware/drivers/rtc/rtc_ds1339_ds3231.c

160 lines
4.4 KiB
C
Raw Normal View History

/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
* $Id$
*
* Copyright (C) 2006 by Robert Kukla
* based on Archos code by Linus Nielsen Feltzing, Uwe Freese, Laurent Baum
* 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 2
* of the License, or (at your option) any later version.
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
* KIND, either express or implied.
*
****************************************************************************/
#include "rtc.h"
#include "logf.h"
#include "sw_i2c.h"
#include "timefuncs.h"
#define RTC_ADDR 0xD0
void rtc_init(void)
{
sw_i2c_init();
#ifdef HAVE_RTC_ALARM
/* Check + save alarm bit first, before the power thread starts watching */
rtc_check_alarm_started(false);
#endif
}
#ifdef HAVE_RTC_ALARM
/* check whether the unit has been started by the RTC alarm function */
/* (check for A2F, which => started using wakeup alarm) */
bool rtc_check_alarm_started(bool release_alarm)
{
static bool alarm_state, run_before;
bool rc;
if (run_before) {
rc = alarm_state;
alarm_state &= ~release_alarm;
} else {
/* This call resets AF, so we store the state for later recall */
rc = alarm_state = rtc_check_alarm_flag();
run_before = true;
}
return rc;
}
/*
* Checks the A2F flag. This call resets A2F once read.
*
*/
bool rtc_check_alarm_flag(void)
{
unsigned char buf[1];
bool flag = false;
sw_i2c_read(RTC_ADDR, 0x0f, buf, 1);
if (buf[0] & 0x02) flag = true;
buf[0] = 0x00;
sw_i2c_write(RTC_ADDR, 0x0f, buf, 1);
return flag;
}
/* set alarm time registers to the given time (repeat once per day) */
void rtc_set_alarm(int h, int m)
{
unsigned char buf[3];
buf[0] = (((m / 10) << 4) | (m % 10)) & 0x7f; /* minutes */
buf[1] = (((h / 10) << 4) | (h % 10)) & 0x3f; /* hour */
buf[2] = 0x80; /* repeat every day */
sw_i2c_write(RTC_ADDR, 0x0b, buf, 3);
}
/* read out the current alarm time */
void rtc_get_alarm(int *h, int *m)
{
unsigned char buf[2];
sw_i2c_read(RTC_ADDR, 0x0b, buf, 2);
*m = ((buf[0] & 0x70) >> 4) * 10 + (buf[0] & 0x0f);
*h = ((buf[1] & 0x30) >> 4) * 10 + (buf[1] & 0x0f);
}
/* turn alarm on or off by setting the alarm flag enable */
/* the alarm is automatically disabled when the RTC gets Vcc power at startup */
/* avoid that an alarm occurs when the device is on because this locks the ON key forever */
void rtc_enable_alarm(bool enable)
{
unsigned char buf[2];
buf[0] = enable ? 0x26 : 0x24; /* BBSQI INTCN A2IE vs INTCH only */
buf[1] = 0x00; /* reset alarm flags (and OSF for good measure) */
sw_i2c_write(RTC_ADDR, 0x0e, buf, 2);
}
#endif /* HAVE_RTC_ALARM */
int rtc_read_datetime(struct tm *tm)
{
int rc;
unsigned char buf[7];
rc = sw_i2c_read(RTC_ADDR, 0, buf, sizeof(buf));
/* convert from bcd, avoid getting extra bits */
tm->tm_sec = BCD2DEC(buf[0] & 0x7f);
tm->tm_min = BCD2DEC(buf[1] & 0x7f);
tm->tm_hour = BCD2DEC(buf[2] & 0x3f);
tm->tm_mday = BCD2DEC(buf[4] & 0x3f);
tm->tm_mon = BCD2DEC(buf[5] & 0x1f) - 1;
tm->tm_year = BCD2DEC(buf[6]) + 100;
set_day_of_week(tm);
set_day_of_year(tm);
return rc;
}
int rtc_write_datetime(const struct tm *tm)
{
unsigned int i;
int rc;
unsigned char buf[7];
buf[0] = tm->tm_sec;
buf[1] = tm->tm_min;
buf[2] = tm->tm_hour;
buf[3] = tm->tm_wday + 1; /* chip wants 1..7 for wday */
buf[4] = tm->tm_mday;
buf[5] = tm->tm_mon + 1;
buf[6] = tm->tm_year - 100;
for (i = 0; i < sizeof(buf); i++)
buf[i] = DEC2BCD(buf[i]);
buf[5] |= 0x80; /* chip wants century (always 20xx) */
rc = sw_i2c_write(RTC_ADDR, 0, buf, sizeof(buf));
return rc;
}