Solution how to fix broken RTC on Nucleo_F103RB / STM32F103 BluePill etc..
Fork of Nucleo_RTC_battery_bkup_pwr_off_okay by
Experimental fork https://os.mbed.com/users/kenjiArai/code/Nucleo_RTC_battery_bkup_pwr_off_okay/ to fix broken RTC on Nucleo_F103RB / STM32F103 BluePill etc..
At this moment (7/11/17) use forked mbed-dev https://os.mbed.com/users/maxxir/code/mbed-dev/.
Or require patch for ./mbed-dev/targets/TARGET_STM/rtc_api.c.
You can manual add to your project fresh mbed-dev and change file mbed-dev\targets\TARGET_STM\rtc_api.c from root project patched rtc_api.c.stm32f10x.txt.
Exploring origin errors, I saw that something wrong with HAL API on STM32F1xx with this functions:
HAL_RTC_GetTime(&RtcHandle, &timeStruct, RTC_FORMAT_BIN); HAL_RTC_GetDate(&RtcHandle, &dateStruct, RTC_FORMAT_BIN); HAL_RTC_SetDate(&RtcHandle, &dateStruct, RTC_FORMAT_BIN); HAL_RTC_SetTime(&RtcHandle, &timeStruct, RTC_FORMAT_BIN);
Look here (as I understand it possible broken on STM32CUBE HAL level now):
https://community.st.com/thread/43218-stm32f103-loss-rtc-date-when-reset
So I use direct RTC register manipulation for STM32F1xx:
rtc_read(), rtc_write() (native rtc_init() - works good).
Also added stub for non-working on STM32F1xx rtc_read_subseconds().
Now the stm32F103 can survive power off, and allows you to get and set the time.
Tested OK on boards:
NUCLEO STM32F103RB, DIY STM32F100CB (forked from DISCO_F100RB)
Happy coding!
maxxir
10/11/17
main.cpp
- Committer:
- kenjiArai
- Date:
- 2015-02-14
- Revision:
- 1:7a48c475bbd2
- Parent:
- 0:0751c92c0f71
- Child:
- 2:77ec44bf1fa5
File content as of revision 1:7a48c475bbd2:
/* * mbed Application program * RTC (inside STM32x CPU) test program * * Copyright (c) 2015 Kenji Arai / JH1PJL * http://www.page.sannet.ne.jp/kenjia/index.html * http://mbed.org/users/kenjiArai/ * Created: January 17th, 2015 * Revised: Feburary 14th, 2015 * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, * INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE * AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ // Include --------------------------------------------------------------------------------------- #include "mbed.h" #include "TextLCD.h" // MUST -> need to modify rtc_api.c, please refer SetRTC.h file [CATION] and modify_info_xxx.h #include "SetRTC.h" // Definition ------------------------------------------------------------------------------------ #define SHOW_KEY_PROMPT 30 #define USE_LCD #if defined(TARGET_NUCLEO_F401RE) || defined(TARGET_NUCLEO_F411RE) || defined(TARGET_NUCLEO_L152RE) #else #error "Target is only Nucleo F401RE, F411RE and L152RE" #endif // Object ---------------------------------------------------------------------------------------- DigitalIn userSW(USER_BUTTON); DigitalOut myled(LED1); // Indicate the sampling period Serial pc(USBTX, USBRX); #if defined(USE_LCD) I2C i2c(D14,D15); // SDA, SCL TextLCD_I2C_N lcd(&i2c, 0x7c, TextLCD::LCD8x2); // LCD(Akizuki AQM0802A) #endif // RAM ------------------------------------------------------------------------------------------- // ROM / Constant data --------------------------------------------------------------------------- // Function prototypes --------------------------------------------------------------------------- //------------------------------------------------------------------------------------------------- // Control Program //------------------------------------------------------------------------------------------------- int main() { char buf[42]; // data buffer for text time_t seconds; uint8_t counter = SHOW_KEY_PROMPT; #if defined(USE_LCD) // lcd lcd.locate(0, 0); // 1st line top // 12345678 lcd.printf(" RTC "); lcd.locate(0, 1); // 2nd line top // 12345678 lcd.puts(" JH1PJL "); lcd.setContrast(0x14); #endif pc.printf("\r\n\r\nTest Nucleo RTC Function\r\n"); // waiting for Initial screen myled = 1; wait(1.0); myled = !myled; wait(1.0); myled = !myled; if (SetRTC() == OK) { pc.printf("External Xtal for RTC\r\n"); } else { pc.printf("Internal Xtal for RTC\r\n"); } show_RTC_reg(); // only for debug purpose while(1) { myled = 1; seconds = time(NULL); strftime(buf, 40, " %B %d,'%y, %H:%M:%S\r\n", localtime(&seconds)); pc.printf("[Time] %s", buf); #if defined(USE_LCD) lcd.locate(0, 0); // 1st line top strftime(buf, 40, "%b%d'%y", localtime(&seconds)); lcd.printf(buf); lcd.locate(0, 1); // 2nd line top strftime(buf, 10, "%H:%M:%S", localtime(&seconds)); lcd.printf(buf); #endif if (counter){ --counter; // 012345678901234567890123456789012345678901234567890123456789012 pc.printf("Is time correct? If no, hit any key.\r"); wait(1.0); if (pc.readable() == 1){ buf[0] = pc.getc(); // dummy read time_enter_mode(); counter = SHOW_KEY_PROMPT; } pc.printf(" \r"); } else { wait(1.0); } if (userSW == 0){ #if defined(USE_LCD) lcd.locate(0, 0); // 1st line top // 12345678 lcd.printf(" Enter "); lcd.locate(0, 1); // 2nd line top // 12345678 lcd.printf(" Sleep " ); #endif pc.printf("\r\nEnter Deep Sleep Mode, please push RESET to wake-up\r\n"); wait(1.0); myled = 0; deepsleep_preparation(); deepsleep(); } } }