Skip to content

Commit

Permalink
Add a 64bit version of the millis() function
Browse files Browse the repository at this point in the history
millis() will rollover after 50 days leading to some interesting bugs.
Create a 64bit version.

Signed-off-by: jgstroud <[email protected]>
  • Loading branch information
jgstroud committed Dec 28, 2024
1 parent 216fa13 commit 1220608
Show file tree
Hide file tree
Showing 9 changed files with 60 additions and 55 deletions.
2 changes: 1 addition & 1 deletion platformio.ini
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ build_flags =
-Llib/lwip2
-llwip2-536
${env.build_flags}
'-Wno-unused-variable'
'-Wall'
-D PIO_FRAMEWORK_ARDUINO_LWIP2_LOW_MEMORY_LOW_FLASH
-D PIO_FRAMEWORK_ARDUINO_MMU_CACHE16_IRAM48_SECHEAP_SHARED
-D LOG_MSG_BUFFER
Expand Down
52 changes: 26 additions & 26 deletions src/comms.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -52,8 +52,8 @@ uint32_t last_saved_code = 0;

static const uint8_t RX_LENGTH = 2;
typedef uint8_t RxPacket[RX_LENGTH * 4];
unsigned long last_rx;
unsigned long last_tx;
uint64_t last_rx;
uint64_t last_tx;

bool wallplateBooting = false;
bool wallPanelDetected = false;
Expand Down Expand Up @@ -190,10 +190,10 @@ void wallPlate_Emulation()
if (wallPanelDetected)
return;

unsigned long currentMillis = millis();
static unsigned long lastRequestMillis = 0;
uint64_t currentMillis = millis64();
static uint64_t lastRequestMillis = 0;
static bool emulateWallPanel = false;
static unsigned long serialDetected = 0;
static uint64_t serialDetected = 0;
static uint8_t stateIndex = 0;

if (!serialDetected)
Expand Down Expand Up @@ -268,7 +268,7 @@ void comms_loop_sec1()
if (sw_serial.available())
{
uint8_t ser_byte = sw_serial.read();
last_rx = millis();
last_rx = millis64();

if (!reading_msg)
{
Expand Down Expand Up @@ -304,7 +304,7 @@ void comms_loop_sec1()
gotMessage = true;
}

if (gotMessage == false && (millis() - last_rx) > 100)
if (gotMessage == false && (millis64() - last_rx) > 100)
{
RINFO("RX message timeout");
// if we have a partial packet and it's been over 100ms since last byte was read,
Expand Down Expand Up @@ -332,7 +332,7 @@ void comms_loop_sec1()
manual_recovery();
if (motionTriggers.bit.doorKey)
{
garage_door.motion_timer = millis() + 5000;
garage_door.motion_timer = millis64() + 5000;
garage_door.motion = true;
notify_homekit_motion();
}
Expand Down Expand Up @@ -548,7 +548,7 @@ void comms_loop_sec1()
notify_homekit_light();
if (motionTriggers.bit.lightKey)
{
garage_door.motion_timer = millis() + 5000;
garage_door.motion_timer = millis64() + 5000;
garage_door.motion = true;
notify_homekit_motion();
}
Expand Down Expand Up @@ -576,7 +576,7 @@ void comms_loop_sec1()
notify_homekit_current_lock();
if (motionTriggers.bit.lockKey)
{
garage_door.motion_timer = millis() + 5000;
garage_door.motion_timer = millis64() + 5000;
garage_door.motion = true;
notify_homekit_motion();
}
Expand All @@ -592,13 +592,13 @@ void comms_loop_sec1()
//
PacketAction pkt_ac;
static unsigned long cmdDelay = 0;
unsigned long now;
uint64_t now;
bool okToSend = false;

if (!q_isEmpty(&pkt_q))
{

now = millis();
now = millis64();

// if there is no wall panel, no need to check 200ms since last rx
// (yes some duped code here, but its clearer)
Expand Down Expand Up @@ -798,7 +798,7 @@ void comms_loop_sec2()
notify_homekit_target_lock();
if (motionTriggers.bit.lockKey)
{
garage_door.motion_timer = millis() + 5000;
garage_door.motion_timer = millis64() + 5000;
garage_door.motion = true;
notify_homekit_motion();
}
Expand Down Expand Up @@ -832,7 +832,7 @@ void comms_loop_sec2()
notify_homekit_light();
if (motionTriggers.bit.lightKey)
{
garage_door.motion_timer = millis() + 5000;
garage_door.motion_timer = millis64() + 5000;
garage_door.motion = true;
notify_homekit_motion();
}
Expand Down Expand Up @@ -863,7 +863,7 @@ void comms_loop_sec2()
/* When we get the motion detect message, notify HomeKit. Motion sensor
will continue to send motion messages every 5s until motion stops.
set a timer for 5 seconds to disable motion after the last message */
garage_door.motion_timer = millis() + 5000;
garage_door.motion_timer = millis64() + 5000;
if (!garage_door.motion)
{
garage_door.motion = true;
Expand All @@ -883,7 +883,7 @@ void comms_loop_sec2()
}
if (pkt.m_data.value.door_action.pressed && motionTriggers.bit.doorKey)
{
garage_door.motion_timer = millis() + 5000;
garage_door.motion_timer = millis64() + 5000;
garage_door.motion = true;
notify_homekit_motion();
}
Expand Down Expand Up @@ -980,7 +980,7 @@ bool transmitSec1(byte toSend)
}

sw_serial.write(toSend);
last_tx = millis();
last_tx = millis64();

// RINFO("SEC1 SEND BYTE: %02X",toSend);

Expand Down Expand Up @@ -1054,7 +1054,7 @@ bool process_PacketAction(PacketAction &pkt_ac)
success = transmitSec1(pkt_ac.pkt.m_data.value.cmd);
if (success)
{
last_tx = millis();
last_tx = millis64();
// RINFO("sending 0x%02X query", pkt_ac.pkt.m_data.value.cmd);
}
}
Expand All @@ -1067,7 +1067,7 @@ bool process_PacketAction(PacketAction &pkt_ac)
success = transmitSec1(secplus1Codes::DoorButtonPress);
if (success)
{
last_tx = millis();
last_tx = millis64();
RINFO("sending DOOR button press");
}
}
Expand All @@ -1076,7 +1076,7 @@ bool process_PacketAction(PacketAction &pkt_ac)
success = transmitSec1(secplus1Codes::DoorButtonRelease);
if (success)
{
last_tx = millis();
last_tx = millis64();
RINFO("sending DOOR button release");
}
}
Expand All @@ -1091,7 +1091,7 @@ bool process_PacketAction(PacketAction &pkt_ac)
success = transmitSec1(secplus1Codes::LightButtonPress);
if (success)
{
last_tx = millis();
last_tx = millis64();
RINFO("sending LIGHT button press");
}
}
Expand All @@ -1100,7 +1100,7 @@ bool process_PacketAction(PacketAction &pkt_ac)
success = transmitSec1(secplus1Codes::LightButtonRelease);
if (success)
{
last_tx = millis();
last_tx = millis64();
RINFO("Sending LIGHT button release");
}
}
Expand All @@ -1115,7 +1115,7 @@ bool process_PacketAction(PacketAction &pkt_ac)
success = transmitSec1(secplus1Codes::LockButtonPress);
if (success)
{
last_tx = millis();
last_tx = millis64();
RINFO("sending LOCK button press");
}
}
Expand All @@ -1124,7 +1124,7 @@ bool process_PacketAction(PacketAction &pkt_ac)
success = transmitSec1(secplus1Codes::LockButtonRelease);
if (success)
{
last_tx = millis();
last_tx = millis64();
RINFO("sending LOCK button release");
}
}
Expand Down Expand Up @@ -1452,9 +1452,9 @@ void manual_recovery()
if (force_recover.push_count++ == 0)
{
RINFO("Push count start");
force_recover.timeout = millis() + 3000;
force_recover.timeout = millis64() + 3000;
}
else if (millis() > force_recover.timeout)
else if (millis64() > force_recover.timeout)
{
RINFO("Push count reset");
force_recover.push_count = 0;
Expand Down
2 changes: 1 addition & 1 deletion src/log.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -207,7 +207,7 @@ void printMessageLog(Print &outputDev)
}
#endif
outputDev.write("Server uptime (ms): ");
outputDev.println(millis());
outputDev.println(millis64());
outputDev.write("Firmware version: ");
outputDev.write(AUTO_VERSION);
outputDev.println();
Expand Down
18 changes: 9 additions & 9 deletions src/ratgdo.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -64,8 +64,8 @@ bool previousDryContactDoorClose = false;

struct obstruction_sensor_t
{
unsigned int low_count = 0; // count obstruction low pulses
unsigned long last_asleep = 0; // count time between high pulses from the obst ISR
uint32_t low_count = 0; // count obstruction low pulses
uint64_t last_asleep = 0; // count time between high pulses from the obst ISR
} obstruction_sensor;

// long unsigned int led_reset_time = 0; // Stores time when LED should return to idle state
Expand All @@ -85,10 +85,10 @@ extern "C" uint32_t __crc_val;
// Track our memory usage
uint32_t free_heap = 65535;
uint32_t min_heap = 65535;
unsigned long next_heap_check = 0;
uint64_t next_heap_check = 0;

bool status_done = false;
unsigned long status_timeout;
uint64_t status_timeout;

/********************************** MAIN LOOP CODE *****************************************/

Expand Down Expand Up @@ -267,8 +267,8 @@ void IRAM_ATTR isr_obstruction()

void obstruction_timer()
{
unsigned long current_millis = millis();
static unsigned long last_millis = 0;
uint64_t current_millis = millis64();
static uint64_t last_millis = 0;

// the obstruction sensor has 3 states: clear (HIGH with LOW pulse every 7ms), obstructed (HIGH), asleep (LOW)
// the transitions between awake and asleep are tricky because the voltage drops slowly when falling asleep
Expand Down Expand Up @@ -338,7 +338,7 @@ void service_timer_loop()
// Service the Obstruction Timer
obstruction_timer();

unsigned long current_millis = millis();
uint64_t current_millis = millis64();

#ifdef NTP_CLIENT
if (enableNTP && clockSet && lastRebootAt == 0)
Expand Down Expand Up @@ -419,9 +419,9 @@ void LED::flash(unsigned long ms)
if (ms)
{
digitalWrite(LED_BUILTIN, activeState);
resetTime = millis() + ms;
resetTime = millis64() + ms;
}
else if ((digitalRead(LED_BUILTIN) == activeState) && (millis() > resetTime))
else if ((digitalRead(LED_BUILTIN) == activeState) && (millis64() > resetTime))
{
digitalWrite(LED_BUILTIN, idleState);
}
Expand Down
6 changes: 3 additions & 3 deletions src/ratgdo.h
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,7 @@ struct GarageDoor
GarageDoorTargetState target_state;
bool obstructed;
bool has_motion_sensor;
unsigned long motion_timer;
uint64_t motion_timer;
bool motion;
bool light;
LockCurrentState current_lock;
Expand All @@ -70,15 +70,15 @@ struct GarageDoor
struct ForceRecover
{
uint8_t push_count;
unsigned long timeout;
uint64_t timeout;
};

class LED
{
private:
uint8_t activeState = 0;
uint8_t idleState = 1; // opposite of active
unsigned long resetTime = 0; // Stores time when LED should return to idle state
uint64_t resetTime = 0; // Stores time when LED should return to idle state
bool initialized = false;
bool enabled = true;

Expand Down
8 changes: 6 additions & 2 deletions src/utilities.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -49,9 +49,13 @@ bool syslogEn = false;
#ifdef NTP_CLIENT
bool clockSet = false;
bool enableNTP = false;
unsigned long lastRebootAt = 0;
uint64_t lastRebootAt = 0;
int32_t savedDoorUpdateAt = 0;

uint64_t IRAM_ATTR millis64() {
return (micros64() / 1000);
}

bool get_tz()
{
WiFiClient client;
Expand Down Expand Up @@ -529,4 +533,4 @@ bool read_config_from_file()
}
file.close();
return true;
}
}
3 changes: 2 additions & 1 deletion src/utilities.h
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@

#ifdef NTP_CLIENT
extern bool clockSet;
extern unsigned long lastRebootAt;
extern uint64_t lastRebootAt;
extern char *timeString(time_t reqTime = 0, bool syslog = false);
extern bool enableNTP;
#define NTP_SERVER "pool.ntp.org"
Expand Down Expand Up @@ -97,6 +97,7 @@ typedef union
extern motionTriggersUnion motionTriggers;

// Function declarations
uint64_t millis64();
void load_all_config_settings();
void sync_and_restart();

Expand Down
6 changes: 3 additions & 3 deletions src/web.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -266,7 +266,7 @@ char *json = NULL;
void web_loop()
{
loop_id = LOOP_WEB;
unsigned long upTime = millis();
uint64_t upTime = millis64();
START_JSON(json);
if (garage_door.active && garage_door.current_state != lastDoorState)
{
Expand Down Expand Up @@ -590,7 +590,7 @@ void handle_everything()

void handle_status()
{
unsigned long upTime = millis();
uint64_t upTime = millis64();
#define paired homekit_is_paired()
#define accessoryID arduino_homekit_get_running_server() ? arduino_homekit_get_running_server()->accessory_id : "Inactive"
#define clientCount arduino_homekit_get_running_server() ? arduino_homekit_get_running_server()->nfds : 0
Expand Down Expand Up @@ -1016,7 +1016,7 @@ void SSEheartbeat(SSESubscription *s)
static int lastClientCount = 0;

START_JSON(json);
ADD_INT(json, "upTime", millis());
ADD_INT(json, "upTime", millis64());
ADD_INT(json, "freeHeap", free_heap);
ADD_INT(json, "minHeap", min_heap);
ADD_INT(json, "minStack", ESP.getFreeContStack());
Expand Down
Loading

0 comments on commit 1220608

Please sign in to comment.