This Arduino sketch is built based on Stm32duino libmaple core. It provides an example of how one could adjust for RTC (real time clock) drifts if one is using a 32k crystal that drift significantly.
For drifts that is less than (304 secs ~ 5 minutes) per month. STM has published an appnote for a different means of slowing down the RTC clock AN2604 STM32F101xx and STM32F103xx RTC calibration. This is possibly simpler and more accurate. this feature has been added in this implementation
Due to the use of backup registers, you need to power the board/stm32 on VBAT (e.g. using a coin cell) so that the backup memory is maintained. And as the last adjusted date/time (saved in backup register 8 and 9) and drift duration (saved in backup register 7), if power is removed and the backup memory is lost, you would need to re-do the calibration again.
Build and run the sketch based on STM32duino libmaple core using the Arduino IDE. The files relevant to the sketch are:
- RTCAdjust.ino - this is the sketch itself
- rtadjust.h - utility functions for the adjustments
- rtadjust.cpp - utility functions for the adjustments
The sketch is a demo that runs on the serial console, the commands on the serial console are
- h, H, ? - display help this message
- t - display current time
- sYYYY-MM-DD HH:MM:SS - sync/set time
- cYYYY-MM-DD HH:MM:SS - calibrate rtc
where YYYY-MM-DD HH:MM_SS - is the current accurate time - xnnnn - set the drift duration where nnnn is the drift duration
number of seconds for the stm32 rtc to drift 1 secs (faster)
t command displays the time and some additional info related to the adjustments e.g.
RTC timestamp: 2018-12-10, 00:00:01
last adj:RTC timestamp: 2018-12-10, 00:00:00
drift duration, number of seconds for the stm32 rtc to drift 1 secs (faster): 3350
In your wiring/arduino setup() function, make it call the function adjtime();, e.g.
void setup() {
/* initialise access to backup registers,
* this is necessary due to the use of backup registers */
bkp_init();
/* adjust rtc */
adjtime();
}
This function would make RTC adjustments based a calibrated drift duration or offset which i named as the drift duration. The drift duration is number of seconds for the stm32 rtc to drift 1 seconds (faster)
This app/sketch itself has the commands to do the calibration. To do the calibration:
-
run sYYYY-MM-DD HH:MM:SS - sync/set time
where YYYY-MM-DD HH:MM_SS - is the current accurate date/time This would set the RTC and the last adjustment date/time to the date / time you supplied.
The last adjustment date/time is saved in 2 backup registers 8 and 9. -
after a day or more, a longer period cumulates more drift and increase accuracy
run cYYYY-MM-DD HH:MM:SS - calibrate rtc
where YYYY-MM-DD HH:MM_SS - is the current accurate date/time at the time you do the calibration.This would compute the drift duration and save it in backup register 7.
-
reset the device and check the time after the calibration. if setup() calls adjtime(), the RTC would show the drift adjusted/compensated date/time.
- in setup() run adjtime()
void setup() {
/* initialise access to backup registers,
* this is necessary due to the use of backup registers */
bkp_init();
/* adjust rtc */
adjtime();
}
This would make the adjustments every time the sketch is started. Note that adjtime() does the adjustment no more than once in 24 hours. This design is to prevent frequent adjustments which could result in cumulative errors as the granuality of adjustments is 1 sec. if you prefer for the adjustments to be more frequent you could comment the statement in the function.
- The function to set the RTC and last adjusted date time is:
/* synctime() set the RTC clock and saves the time in backup register 8 and 9
* as the time of the last adjustment
*
* call this function with the current accurate clock time to set the rtc
*
* @param time the time_t value of the current accurate clock time
*/
void synctime(time_t time_now);
- The function to perform the RTC calibration to compute the drift duration is
/* this function calibrate the rtc by computing the drift_duration
*
* call this function with the current accurate clock time to calibrate the rtc
*
* if the cumulative delay between current time and the last time when synctime()
* is called is lower than 100, a warning would be displayed that the drift
* granulity is low and may result in inaccuracy of the rtc adjustments
*
* note that this function can only be run once to compute the drift_duration
* this is because the time of last adjustment would have been updated
* by adjtime() after calibration and is no longer relevant for purpose of
* computing drift duration
*
* to run it again
* 1) first zero out drift duration using setbkpdrift(0)
* or disconnect VBAT and power to clear backup memory
* next repeat the calibration cycle
* 2) run synctime(time_now) with the accurate clock time
* 3) after a day or more (longer period cumulates more drift and increase accuracy)
* run calibrate(time_now) with the accurate clock time
*
* @param time the time_t value of the current accurate clock time
*/
void calibratertc(time_t time_now);
- Dependencies, in the main sketch:
/* note that the adjustment functions needs this RTClock
* if you rename variable rt, update rtadjust.h so that the
* extern refers to this RTClock*/
RTClock rt(RTCSEL_LSE); // initialise