296 lines
8.9 KiB
C++
296 lines
8.9 KiB
C++
|
|
/*
|
||
|
|
TimeAlarms.cpp - Arduino Time alarms for use with Time library
|
||
|
|
Copyright (c) 2008-2011 Michael Margolis.
|
||
|
|
|
||
|
|
This library is free software; you can redistribute it and/or
|
||
|
|
modify it under the terms of the GNU Lesser General Public
|
||
|
|
License as published by the Free Software Foundation; either
|
||
|
|
version 2.1 of the License, or (at your option) any later version.
|
||
|
|
|
||
|
|
This library 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
|
||
|
|
Lesser General Public License for more details.
|
||
|
|
*/
|
||
|
|
|
||
|
|
/*
|
||
|
|
2 July 2011 - replaced alarm types implied from alarm value with enums to make trigger logic more robust
|
||
|
|
- this fixes bug in repeating weekly alarms - thanks to Vincent Valdy and draythomp for testing
|
||
|
|
*/
|
||
|
|
|
||
|
|
#include "ESP8266TimeAlarms.h"
|
||
|
|
|
||
|
|
#define IS_ONESHOT true // constants used in arguments to create method
|
||
|
|
#define IS_REPEAT false
|
||
|
|
|
||
|
|
|
||
|
|
//**************************************************************
|
||
|
|
//* Alarm Class Constructor
|
||
|
|
|
||
|
|
AlarmClass::AlarmClass()
|
||
|
|
{
|
||
|
|
Mode.isEnabled = Mode.isOneShot = 0;
|
||
|
|
Mode.alarmType = dtNotAllocated;
|
||
|
|
value = nextTrigger = 0;
|
||
|
|
onTickHandler = NULL; // prevent a callback until this pointer is explicitly set
|
||
|
|
}
|
||
|
|
|
||
|
|
//**************************************************************
|
||
|
|
//* Private Methods
|
||
|
|
|
||
|
|
|
||
|
|
void AlarmClass::updateNextTrigger()
|
||
|
|
{
|
||
|
|
if (Mode.isEnabled) {
|
||
|
|
time_t now = time(nullptr);
|
||
|
|
|
||
|
|
if (dtIsAlarm(Mode.alarmType) && nextTrigger <= now) {
|
||
|
|
// update alarm if next trigger is not yet in the future
|
||
|
|
if (Mode.alarmType == dtExplicitAlarm) {
|
||
|
|
// is the value a specific date and time in the future
|
||
|
|
nextTrigger = value; // yes, trigger on this value
|
||
|
|
} else if (Mode.alarmType == dtDailyAlarm) {
|
||
|
|
//if this is a daily alarm
|
||
|
|
if (value + previousMidnight(now) <= now) {
|
||
|
|
// if time has passed then set for tomorrow
|
||
|
|
nextTrigger = value + nextMidnight(now);
|
||
|
|
} else {
|
||
|
|
// set the date to today and add the time given in value
|
||
|
|
nextTrigger = value + previousMidnight(now);
|
||
|
|
}
|
||
|
|
} else if (Mode.alarmType == dtWeeklyAlarm) {
|
||
|
|
// if this is a weekly alarm
|
||
|
|
if ((value + previousSunday(now)) <= now) {
|
||
|
|
// if day has passed then set for the next week.
|
||
|
|
nextTrigger = value + nextSunday(now);
|
||
|
|
} else {
|
||
|
|
// set the date to this week today and add the time given in value
|
||
|
|
nextTrigger = value + previousSunday(now);
|
||
|
|
}
|
||
|
|
} else {
|
||
|
|
// its not a recognized alarm type - this should not happen
|
||
|
|
Mode.isEnabled = false; // Disable the alarm
|
||
|
|
}
|
||
|
|
}
|
||
|
|
if (Mode.alarmType == dtTimer) {
|
||
|
|
// its a timer
|
||
|
|
nextTrigger = now + value; // add the value to previous time (this ensures delay always at least Value seconds)
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
//**************************************************************
|
||
|
|
//* Time Alarms Public Methods
|
||
|
|
|
||
|
|
TimeAlarmsClass::TimeAlarmsClass()
|
||
|
|
{
|
||
|
|
isServicing = false;
|
||
|
|
for(uint8_t id = 0; id < dtNBR_ALARMS; id++) {
|
||
|
|
free(id); // ensure all Alarms are cleared and available for allocation
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
void TimeAlarmsClass::enable(AlarmID_t ID)
|
||
|
|
{
|
||
|
|
if (isAllocated(ID)) {
|
||
|
|
if (( !(dtUseAbsoluteValue(Alarm[ID].Mode.alarmType) && (Alarm[ID].value == 0)) ) && (Alarm[ID].onTickHandler != NULL)) {
|
||
|
|
// only enable if value is non zero and a tick handler has been set
|
||
|
|
// (is not NULL, value is non zero ONLY for dtTimer & dtExplicitAlarm
|
||
|
|
// (the rest can have 0 to account for midnight))
|
||
|
|
Alarm[ID].Mode.isEnabled = true;
|
||
|
|
Alarm[ID].updateNextTrigger(); // trigger is updated whenever this is called, even if already enabled
|
||
|
|
} else {
|
||
|
|
Alarm[ID].Mode.isEnabled = false;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
void TimeAlarmsClass::disable(AlarmID_t ID)
|
||
|
|
{
|
||
|
|
if (isAllocated(ID)) {
|
||
|
|
Alarm[ID].Mode.isEnabled = false;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
// write the given value to the given alarm
|
||
|
|
void TimeAlarmsClass::write(AlarmID_t ID, time_t value)
|
||
|
|
{
|
||
|
|
if (isAllocated(ID)) {
|
||
|
|
Alarm[ID].value = value; //note: we don't check value as we do it in enable()
|
||
|
|
Alarm[ID].nextTrigger = 0; // clear out previous trigger time (see issue #12)
|
||
|
|
enable(ID); // update trigger time
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
// return the value for the given alarm ID
|
||
|
|
time_t TimeAlarmsClass::read(AlarmID_t ID)
|
||
|
|
{
|
||
|
|
if (isAllocated(ID)) {
|
||
|
|
return Alarm[ID].value ;
|
||
|
|
} else {
|
||
|
|
return dtINVALID_TIME;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
// return the alarm type for the given alarm ID
|
||
|
|
dtAlarmPeriod_t TimeAlarmsClass::readType(AlarmID_t ID)
|
||
|
|
{
|
||
|
|
if (isAllocated(ID)) {
|
||
|
|
return (dtAlarmPeriod_t)Alarm[ID].Mode.alarmType ;
|
||
|
|
} else {
|
||
|
|
return dtNotAllocated;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
void TimeAlarmsClass::free(AlarmID_t ID)
|
||
|
|
{
|
||
|
|
if (isAllocated(ID)) {
|
||
|
|
Alarm[ID].Mode.isEnabled = false;
|
||
|
|
Alarm[ID].Mode.alarmType = dtNotAllocated;
|
||
|
|
Alarm[ID].onTickHandler = NULL;
|
||
|
|
Alarm[ID].value = 0;
|
||
|
|
Alarm[ID].nextTrigger = 0;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
// returns the number of allocated timers
|
||
|
|
uint8_t TimeAlarmsClass::count()
|
||
|
|
{
|
||
|
|
uint8_t c = 0;
|
||
|
|
for(uint8_t id = 0; id < dtNBR_ALARMS; id++) {
|
||
|
|
if (isAllocated(id)) c++;
|
||
|
|
}
|
||
|
|
return c;
|
||
|
|
}
|
||
|
|
|
||
|
|
// returns true only if id is allocated and the type is a time based alarm, returns false if not allocated or if its a timer
|
||
|
|
bool TimeAlarmsClass::isAlarm(AlarmID_t ID)
|
||
|
|
{
|
||
|
|
return( isAllocated(ID) && dtIsAlarm(Alarm[ID].Mode.alarmType) );
|
||
|
|
}
|
||
|
|
|
||
|
|
// returns true if this id is allocated
|
||
|
|
bool TimeAlarmsClass::isAllocated(AlarmID_t ID)
|
||
|
|
{
|
||
|
|
return (ID < dtNBR_ALARMS && Alarm[ID].Mode.alarmType != dtNotAllocated);
|
||
|
|
}
|
||
|
|
|
||
|
|
// returns the currently triggered alarm id
|
||
|
|
// returns dtINVALID_ALARM_ID if not invoked from within an alarm handler
|
||
|
|
AlarmID_t TimeAlarmsClass::getTriggeredAlarmId()
|
||
|
|
{
|
||
|
|
if (isServicing) {
|
||
|
|
return servicedAlarmId; // new private data member used instead of local loop variable i in serviceAlarms();
|
||
|
|
} else {
|
||
|
|
return dtINVALID_ALARM_ID; // valid ids only available when servicing a callback
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
// following functions are not Alarm ID specific.
|
||
|
|
void TimeAlarmsClass::delay(unsigned long ms)
|
||
|
|
{
|
||
|
|
unsigned long start = millis();
|
||
|
|
while (millis() - start <= ms) {
|
||
|
|
serviceAlarms();
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
void TimeAlarmsClass::waitForDigits( uint8_t Digits, dtUnits_t Units)
|
||
|
|
{
|
||
|
|
while (Digits != getDigitsNow(Units)) {
|
||
|
|
serviceAlarms();
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
void TimeAlarmsClass::waitForRollover( dtUnits_t Units)
|
||
|
|
{
|
||
|
|
// if its just rolled over than wait for another rollover
|
||
|
|
while (getDigitsNow(Units) == 0) {
|
||
|
|
serviceAlarms();
|
||
|
|
}
|
||
|
|
waitForDigits(0, Units);
|
||
|
|
}
|
||
|
|
|
||
|
|
uint8_t TimeAlarmsClass::getDigitsNow( dtUnits_t Units)
|
||
|
|
{
|
||
|
|
time_t now = time(nullptr);
|
||
|
|
struct tm *NOW = localtime(&now);
|
||
|
|
|
||
|
|
if (Units == dtSecond) return NOW->tm_sec;
|
||
|
|
if (Units == dtMinute) return NOW->tm_min;
|
||
|
|
if (Units == dtHour) return NOW->tm_hour;
|
||
|
|
if (Units == dtDay) return NOW->tm_wday;
|
||
|
|
return 255; // This should never happen
|
||
|
|
}
|
||
|
|
|
||
|
|
//returns isServicing
|
||
|
|
bool TimeAlarmsClass::getIsServicing()
|
||
|
|
{
|
||
|
|
return isServicing;
|
||
|
|
}
|
||
|
|
|
||
|
|
//***********************************************************
|
||
|
|
//* Private Methods
|
||
|
|
|
||
|
|
void TimeAlarmsClass::serviceAlarms()
|
||
|
|
{
|
||
|
|
if (!isServicing) {
|
||
|
|
time_t now = time(nullptr);
|
||
|
|
isServicing = true;
|
||
|
|
for (servicedAlarmId = 0; servicedAlarmId < dtNBR_ALARMS; servicedAlarmId++) {
|
||
|
|
if (Alarm[servicedAlarmId].Mode.isEnabled && (now >= Alarm[servicedAlarmId].nextTrigger)) {
|
||
|
|
OnTick_t TickHandler = Alarm[servicedAlarmId].onTickHandler;
|
||
|
|
if (Alarm[servicedAlarmId].Mode.isOneShot) {
|
||
|
|
free(servicedAlarmId); // free the ID if mode is OnShot
|
||
|
|
} else {
|
||
|
|
Alarm[servicedAlarmId].updateNextTrigger();
|
||
|
|
}
|
||
|
|
if (TickHandler != NULL) {
|
||
|
|
TickHandler(); // call the handler
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
isServicing = false;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
// returns the absolute time of the next scheduled alarm, or 0 if none
|
||
|
|
time_t TimeAlarmsClass::getNextTrigger()
|
||
|
|
{
|
||
|
|
time_t nextTrigger = (time_t)0xffffffff; // the max time value
|
||
|
|
|
||
|
|
for (uint8_t id = 0; id < dtNBR_ALARMS; id++) {
|
||
|
|
if (isAllocated(id)) {
|
||
|
|
if (Alarm[id].nextTrigger < nextTrigger) {
|
||
|
|
nextTrigger = Alarm[id].nextTrigger;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
return nextTrigger == (time_t)0xffffffff ? 0 : nextTrigger;
|
||
|
|
}
|
||
|
|
|
||
|
|
// attempt to create an alarm and return true if successful
|
||
|
|
AlarmID_t TimeAlarmsClass::create(time_t value, OnTick_t onTickHandler, uint8_t isOneShot, dtAlarmPeriod_t alarmType)
|
||
|
|
{
|
||
|
|
time_t now = time(nullptr);
|
||
|
|
if ( ! ( (dtIsAlarm(alarmType) && now < SECS_PER_YEAR) || (dtUseAbsoluteValue(alarmType) && (value == 0)) ) ) {
|
||
|
|
// only create alarm ids if the time is at least Jan 1 1971
|
||
|
|
for (uint8_t id = 0; id < dtNBR_ALARMS; id++) {
|
||
|
|
if (Alarm[id].Mode.alarmType == dtNotAllocated) {
|
||
|
|
// here if there is an Alarm id that is not allocated
|
||
|
|
Alarm[id].onTickHandler = onTickHandler;
|
||
|
|
Alarm[id].Mode.isOneShot = isOneShot;
|
||
|
|
Alarm[id].Mode.alarmType = alarmType;
|
||
|
|
Alarm[id].value = value;
|
||
|
|
enable(id);
|
||
|
|
return id; // alarm created ok
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
return dtINVALID_ALARM_ID; // no IDs available or time is invalid
|
||
|
|
}
|
||
|
|
|
||
|
|
// make one instance for the user to use
|
||
|
|
TimeAlarmsClass Alarm = TimeAlarmsClass() ;
|
||
|
|
|