Firmware Reset
During the wifi configuration tutorial I described a method to provide a firmware reset in the case that you forget the user name and password using an additional button (or wire loop). I also lamented the problems of the Wemos D1 board not being able to distinguish between a power cycle and a button reset. Today I'm going to discuss another way to achieve a firmware reset of the device without the requirement for any additional hardware. The method itself is quite simple and we have already used all required parts for the wifi configuration program.
The first thing the device should do when it starts up, is set a flag in the non volatile memory and after a given period of time it should clear this flag again. Now if the device is reset before it has the chance to clear the flag we can tell that the reset button has been pushed twice in quick succession and use that as an indicator that we want to perform a firmware reset.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#include <EEPROM.h> | |
#define FLAGSET 0x55 | |
#define FLAGCLEAR 0xAA | |
#define FLAGADDRESS 00 | |
#define FLAGTIMEOUT 3000 | |
uint32_t flag = FLAGCLEAR; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
bool checkFlag() { | |
flag = EEPROM.read(FLAGADDRESS); | |
return flag == FLAGSET; | |
} | |
void writeFlag(uint32_t value) { | |
flag = value; | |
EEPROM.write(FLAGADDRESS, flag); | |
EEPROM.commit(); | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
void setup() { | |
// put your setup code here, to run once: | |
Serial.begin(115200); Serial.println(); | |
pinMode(LED_BUILTIN, OUTPUT); | |
digitalWrite(LED_BUILTIN, HIGH); | |
//Activate the EEPROM before attempting to use it | |
EEPROM.begin(sizeof(flag)); | |
//Check to see if the flag is still set from the previous boot | |
if(checkFlag()) { | |
//Clear the flag now it has been detected | |
writeFlag(FLAGCLEAR); | |
//Do the firmware reset here | |
Serial.println("Flag Detected"); | |
digitalWrite(LED_BUILTIN, LOW); | |
} | |
else { | |
//Set the flag for detection with the next boot | |
writeFlag(FLAGSET); | |
} | |
// After setup, while away the end of the reset period | |
while (millis() < FLAGTIMEOUT); | |
writeFlag(FLAGCLEAR); | |
Serial.println("Flag Checking Stopped"); | |
} |
Finally there is a little delay loop which waits the given time before clearing the flag, I've added this as a blocking delay at the end of setup because I don't like the idea of my main loop performing a redundant check for the rest of it's execution. It could easily be implemented as a non blocking check in the main loop though.
Now when you download and run the program, if you click the reset button twice within a 3 second period you should see the LED turn on indicating a firmware reset state. As usual the full code is at the bottom.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#include <EEPROM.h> | |
#define FLAGSET 0x55 | |
#define FLAGCLEAR 0xAA | |
#define FLAGADDRESS 00 | |
#define FLAGTIMEOUT 3000 | |
uint32_t flag = FLAGCLEAR; | |
void setup() { | |
// put your setup code here, to run once: | |
Serial.begin(115200); Serial.println(); | |
pinMode(LED_BUILTIN, OUTPUT); | |
digitalWrite(LED_BUILTIN, HIGH); | |
//Activate the EEPROM before attempting to use it | |
EEPROM.begin(sizeof(flag)); | |
//Check to see if the flag is still set from the previous boot | |
if(checkFlag()) { | |
//Clear the flag now it has been detected | |
writeFlag(FLAGCLEAR); | |
//Do the firmware reset here | |
Serial.println("Flag Detected"); | |
digitalWrite(LED_BUILTIN, LOW); | |
} | |
else { | |
//Set the flag for detection with the next boot | |
writeFlag(FLAGSET); | |
} | |
// After setup, while away the end of the reset period | |
while (millis() < FLAGTIMEOUT); | |
writeFlag(FLAGCLEAR); | |
Serial.println("Flag Checking Stopped"); | |
} | |
void loop() { | |
// put your main code here, to run repeatedly: | |
} | |
bool checkFlag() { | |
flag = EEPROM.read(FLAGADDRESS); | |
return flag == FLAGSET; | |
} | |
void writeFlag(uint32_t value) { | |
flag = value; | |
EEPROM.write(FLAGADDRESS, flag); | |
EEPROM.commit(); | |
} |