rockbox/firmware/target/hosted/rtc.c
Solomon Peachy f3010c6a8e hosted: Fix rtc_write_datetime() error handling
(if mktime() fails, don't attempt to write its result to the RTC)

Change-Id: Ib09b8dccafdef032119db76f10af74dfc220cc21
2021-08-05 13:44:11 +00:00

210 lines
4.7 KiB
C

/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
* $Id$
*
* Based upon code (C) 2002 by Björn Stenberg
* Copyright (C) 2011 by Thomas Jarosch
* Copyright (C) 2018 by Marcin Bukat
*
* 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 <time.h>
#include <sys/time.h>
#if !defined(WIN32)
#include <sys/ioctl.h>
#include <linux/rtc.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdbool.h>
#endif
#include "config.h"
void rtc_init(void)
{
tzset();
}
int rtc_read_datetime(struct tm *tm)
{
time_t now = time(NULL);
*tm = *localtime(&now);
return 0;
}
int rtc_write_datetime(const struct tm *tm)
{
#if !defined(WIN32)
struct timeval tv;
struct tm *tm_time;
tv.tv_sec = mktime((struct tm *)tm);
tv.tv_usec = 0;
if ((int)tv.tv_sec == -1)
return -1;
/* set system clock */
settimeofday(&tv, NULL);
/* hw clock stores time in UTC */
time_t now = time(NULL);
tm_time = gmtime(&now);
/* Try to write the HW RTC, if present. */
int rtc = open("/dev/rtc0", O_WRONLY | O_CLOEXEC);
if (rtc > 0) {
ioctl(rtc, RTC_SET_TIME, (struct rtc_time *)tm_time);
close(rtc);
}
#else
(void)(*tm);
#endif
return 0;
}
#if defined(HAVE_RTC_ALARM) && !defined(SIMULATOR)
void rtc_set_alarm(int h, int m)
{
struct rtc_time tm;
long sec;
int rtc = open("/dev/rtc0", O_WRONLY | O_CLOEXEC);
if (rtc < 0)
return;
/* Get RTC time */
ioctl(rtc, RTC_RD_TIME, &tm);
/* Convert to seconds into the GMT day. Can be negative! */
sec = h * 3600 + m * 60 + timezone;
h = sec / 3600;
sec -= h * 3600;
m = sec / 60;
/* Handle negative or positive wraps */
while (m < 0) {
m += 60;
h--;
}
while (m > 59) {
m -= 60;
h++;
}
while (h < 0) {
h += 24;
tm.tm_mday--;
}
while (h > 23) {
h -= 24;
tm.tm_mday++;
}
/* Update the struct */
tm.tm_sec = 0;
tm.tm_hour = h;
tm.tm_min = m;
ioctl(rtc, RTC_ALM_SET, &tm);
close(rtc);
}
void rtc_get_alarm(int *h, int *m)
{
struct rtc_time tm;
long sec;
int rtc = open("/dev/rtc0", O_WRONLY | O_CLOEXEC);
if (rtc < 0)
return;
ioctl(rtc, RTC_ALM_READ, &tm);
close(rtc);
/* Convert RTC from UTC to local time zone.. */
sec = (tm.tm_min * 60) + (tm.tm_hour * 3600) - timezone;
/* Handle wrapping and negative offsets */
*h = (sec / 3600);
sec -= *h * 3600;
*m = sec / 60;
while (*m < 0) {
*m = *m + 60;
*h = *h - 1;
}
while (*m > 59) {
*m = *m - 60;
*h = *h + 1;
}
while (*h < 0) {
*h = *h + 24;
}
while (*h > 23) {
*h = *h - 24;
}
}
void rtc_enable_alarm(bool enable)
{
int rtc = open("/dev/rtc0", O_WRONLY | O_CLOEXEC);
if (rtc < 0)
return;
ioctl(rtc, enable ? RTC_AIE_ON : RTC_AIE_OFF, NULL);
close(rtc);
/* XXX Note that this may or may not work; Linux may need to be suspended
or shut down in a special way to keep the RTC alarm active */
}
/* Returns true if alarm was the reason we started up */
bool rtc_check_alarm_started(bool release_alarm)
{
int rtc = open("/dev/rtc0", O_WRONLY | O_CLOEXEC);
if (rtc < 0)
return false;
/* XXX There is no generic way of determining wakeup reason. Will
likely need a target-specific hook. */
/* Disable alarm if requested */
if (release_alarm)
ioctl(rtc, RTC_AIE_OFF, NULL);
close(rtc);
return false;
}
/* See if we received an alarm. */
bool rtc_check_alarm_flag(void)
{
struct rtc_wkalrm alrm;
int rtc = open("/dev/rtc0", O_WRONLY | O_CLOEXEC);
if (rtc < 0)
return false;
alrm.pending = 0;
/* XXX Documented as "mostly useless on Linux" except with EFI RTCs
Will likely need a target-specific hook. */
ioctl(rtc, RTC_WKALM_RD, &alrm);
close(rtc);
return alrm.pending;
}
#endif /* HAVE_RTC_ALARM */