58b849c451
Some drivers set tm_wday just fine and do not need it coerced to be correct. Others set tm_yday, so don't overwrite what the driver sets; just zero it inside if it can't fill the field. Move calls to set_day_of_week() to the sorts of drivers that presumably required the hammer (FS#11814) in get_time() where the weekday isn't locked to the date. Change-Id: Idd0ded6bfc9d9f48fcc1a6074068164c42fcf24a
159 lines
4.4 KiB
C
159 lines
4.4 KiB
C
/***************************************************************************
|
|
* __________ __ ___.
|
|
* 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;
|
|
tm->tm_yday = 0; /* Not implemented for now */
|
|
|
|
set_day_of_week(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;
|
|
}
|
|
|