// // Created by MightyPork on 2017/07/09. // #include "persist.h" #include <esp8266.h> #include "wifimgr.h" #include "screen.h" PersistBlock persist; #define PERSIST_SECTOR_ID 0x3D //region Persist and restore individual modules static void ICACHE_FLASH_ATTR apply_live_settings(void) { persist_dbg("[Persist] Applying live settings..."); persist_dbg("[Persist] > system"); sysconf_apply_settings(); persist_dbg("[Persist] > wifi"); wifimgr_apply_settings(); persist_dbg("[Persist] > terminal"); terminal_apply_settings(); persist_dbg("[Persist] Live settings applied."); // ... } static void ICACHE_FLASH_ATTR restore_live_settings_to_hard_defaults(void) { persist_dbg("[Persist] Restore to hard defaults..."); persist_dbg("[Persist] > system"); sysconf_restore_defaults(); persist_dbg("[Persist] > wifi"); wifimgr_restore_defaults(); persist_dbg("[Persist] > terminal"); terminal_restore_defaults(); persist_dbg("[Persist] Restored to hard defaults."); // ... } //endregion const u32 wconf_at = (u32)&persist.defaults.wificonf - (u32)&persist.defaults; const u32 tconf_at = (u32)&persist.defaults.termconf - (u32)&persist.defaults; const u32 sconf_at = (u32)&persist.defaults.sysconf - (u32)&persist.defaults; const u32 cksum_at = (u32)&persist.defaults.checksum - (u32)&persist.defaults; /** * Compute CRC32. Adapted from https://github.com/esp8266/Arduino * @param data * @param length * @return crc32 */ static uint32_t ICACHE_FLASH_ATTR calculateCRC32(const uint8_t *data, size_t length) { // the salt here should ensure settings are wiped when the structure changes // CHECKSUM_SALT can be adjusted manually to force a reset. uint32_t crc = 0xffffffff + CHECKSUM_SALT + ((wconf_at << 16) ^ (tconf_at << 10) ^ (sconf_at << 5)); while (length--) { uint8_t c = *data++; for (uint32_t i = 0x80; i > 0; i >>= 1) { bool bit = (bool) (crc & 0x80000000UL); if (c & i) { bit = !bit; } crc <<= 1; if (bit) { crc ^= 0x04c11db7UL; } } } return crc; } /** * Compute a persist bundle checksum * * @param bundle * @return */ static uint32_t ICACHE_FLASH_ATTR compute_checksum(AppConfigBundle *bundle) { // this should be the bundle without the checksum return calculateCRC32((uint8_t *) bundle, sizeof(AppConfigBundle) - 4); } static void ICACHE_FLASH_ATTR set_admin_block_defaults(void) { persist_info("[Persist] Initing admin config block"); strcpy(persist.admin.pw, DEFAULT_ADMIN_PW); persist.admin.version = ADMINCONF_VERSION; } /** * Load, verify and apply persistent config */ void ICACHE_FLASH_ATTR persist_load(void) { persist_info("[Persist] Loading settings from FLASH..."); persist_dbg("Persist memory map:"); persist_dbg("> wifi at %4d (error %2d)", wconf_at, wconf_at - 0); persist_dbg("> sys at %4d (error %2d)", sconf_at, sconf_at - WIFICONF_SIZE); persist_dbg("> term at %4d (error %2d)", tconf_at, tconf_at - WIFICONF_SIZE - SYSCONF_SIZE); persist_dbg("> cksum at %4d (error %2d)", cksum_at, cksum_at - (APPCONF_SIZE - 4)); persist_dbg("> Total size = %d bytes (error %d)", sizeof(AppConfigBundle), APPCONF_SIZE - sizeof(AppConfigBundle)); bool hard_reset = false; // Try to load hard_reset |= !system_param_load(PERSIST_SECTOR_ID, 0, &persist, sizeof(PersistBlock)); // Verify checksums if (hard_reset || (compute_checksum(&persist.defaults) != persist.defaults.checksum) || (compute_checksum(&persist.current) != persist.current.checksum) || (persist.admin.version != 0 && (calculateCRC32((uint8_t *) &persist.admin, sizeof(AdminConfigBlock) - 4) != persist.admin.checksum))) { error("[Persist] Checksum verification: FAILED"); hard_reset = true; } else { persist_info("[Persist] Checksum verification: PASSED"); } if (hard_reset) { // Zero all out memset(&persist, 0, sizeof(PersistBlock)); persist_load_hard_default(); // write them also as defaults memcpy(&persist.defaults, &persist.current, sizeof(AppConfigBundle)); // reset admin pw set_admin_block_defaults(); persist_store(); // this also stores them to flash and applies to modules } else { if (persist.admin.version == 0) { set_admin_block_defaults(); persist_store(); } apply_live_settings(); } persist_info("[Persist] All settings loaded and applied."); } void ICACHE_FLASH_ATTR persist_store(void) { persist_info("[Persist] Storing all settings to FLASH..."); // Update checksums before write persist.current.checksum = compute_checksum(&persist.current); persist.defaults.checksum = compute_checksum(&persist.defaults); persist.admin.checksum = calculateCRC32((uint8_t *) &persist.admin, sizeof(AdminConfigBlock) - 4); if (!system_param_save_with_protect(PERSIST_SECTOR_ID, &persist, sizeof(PersistBlock))) { error("[Persist] Store to flash failed!"); } persist_info("[Persist] All settings persisted."); } /** * Restore to built-in defaults */ void ICACHE_FLASH_ATTR persist_load_hard_default(void) { persist_info("[Persist] Restoring live settings to hard defaults..."); // Set live config to default values restore_live_settings_to_hard_defaults(); persist_store(); persist_info("[Persist] Settings restored to hard defaults."); apply_live_settings(); // apply } /** * Restore default settings & apply. also persists. */ void ICACHE_FLASH_ATTR persist_restore_default(void) { persist_info("[Persist] Restoring live settings to stored defaults..."); memcpy(&persist.current, &persist.defaults, sizeof(AppConfigBundle)); apply_live_settings(); persist_store(); persist_info("[Persist] Settings restored to stored defaults."); } /** * Store current settings as defaults & write to flash */ void ICACHE_FLASH_ATTR persist_set_as_default(void) { persist_info("[Persist] Storing live settings as defaults.."); // current -> defaults memcpy(&persist.defaults, &persist.current, sizeof(AppConfigBundle)); persist_store(); persist_info("[Persist] Default settings updated."); }