/** * @file flash_main.cpp * @brief Higher-level logic of saving data into internal flash memory * * * @date Sep 19, 2013 * @author Andrey Belomutskiy, (c) 2012-2015 */ #include #include "flash_main.h" #include "eficonsole.h" #include "flash.h" #include "rusefi.h" #if EFI_INTERNAL_FLASH #include "engine_controller.h" #include "datalogging.h" #include "engine.h" static bool needToWriteConfiguration = false; EXTERN_ENGINE; static Logging logger; extern persistent_config_container_s persistentState; extern engine_configuration_s *engineConfiguration; #define FLASH_ADDR 0x08060000 #define PERSISTENT_SIZE sizeof(persistent_config_container_s) crc_t flashStateCrc(persistent_config_container_s *state) { return calc_crc((const crc_t*) &state->persistentConfiguration, sizeof(persistent_config_s)); } void setNeedToWriteConfiguration(void) { scheduleMsg(&logger, "Scheduling configuration write"); needToWriteConfiguration = true; } bool getNeedToWriteConfiguration(void) { return needToWriteConfiguration; } void writeToFlashIfPending() { if (!getNeedToWriteConfiguration()) { return; } // todo: technically we need a lock here, realistically we should be fine. needToWriteConfiguration = false; scheduleMsg(&logger, "Writing pending configuration"); writeToFlash(); } extern uint32_t maxLockTime; void writeToFlash(void) { scheduleMsg(&logger, " !!!!!!!!!!!!!!!!!!!! BE SURE NOT WRITE WITH IGNITION ON !!!!!!!!!!!!!!!!!!!!"); persistentState.size = PERSISTENT_SIZE; persistentState.version = FLASH_DATA_VERSION; scheduleMsg(&logger, "flash compatible with %d", persistentState.version); crc_t result = flashStateCrc(&persistentState); persistentState.value = result; scheduleMsg(&logger, "Reseting flash: size=%d", PERSISTENT_SIZE); flashErase(FLASH_ADDR, PERSISTENT_SIZE); scheduleMsg(&logger, "Flashing with CRC=%d", result); efitimems_t nowMs = currentTimeMillis(); result = flashWrite(FLASH_ADDR, (const char *) &persistentState, PERSISTENT_SIZE); scheduleMsg(&logger, "Flash programmed in (ms): %d", currentTimeMillis() - nowMs); scheduleMsg(&logger, "Flashing result: %d", result); maxLockTime = 0; } static bool isValidCrc(persistent_config_container_s *state) { crc_t result = flashStateCrc(state); int isValidCrc_b = result == state->value; if (!isValidCrc_b) { scheduleMsg(&logger, "CRC got %d while %d expected", result, state->value); } return isValidCrc_b; } static void doResetConfiguration(void) { resetConfigurationExt(&logger, engineConfiguration->engineType, engine); } static bool hasValidEngineType(engine_configuration_s *engineConfiguration) { uint32_t ordinal = (uint32_t)engineConfiguration->engineType; return ordinal < ET_UNUSED && engineConfiguration->headerMagicValue == HEADER_MAGIC_NUMBER; } void readFromFlash(void) { printMsg(&logger, "readFromFlash()"); flashRead(FLASH_ADDR, (char *) &persistentState, PERSISTENT_SIZE); if (!isValidCrc(&persistentState)) { printMsg(&logger, "Need to reset flash to default due to CRC"); resetConfigurationExt(&logger, DEFAULT_ENGINE_TYPE, engine); } else if (persistentState.version == FLASH_DATA_VERSION && persistentState.size == PERSISTENT_SIZE) { printMsg(&logger, "Got valid configuration from flash!"); applyNonPersistentConfiguration(&logger, engine); } else if (hasValidEngineType(engineConfiguration)) { printMsg(&logger, "Resetting but saving engine type [%d]", engineConfiguration->engineType); resetConfigurationExt(&logger, engineConfiguration->engineType, engine); } else { printMsg(&logger, "Need to reset flash to default due to version change"); resetConfigurationExt(&logger, DEFAULT_ENGINE_TYPE, engine); } // we can only change the state after the CRC check engineConfiguration->firmwareVersion = getRusEfiVersion(); } static void rewriteConfig(Engine *engine) { doResetConfiguration(); writeToFlash(); } void initFlash(Engine *engine) { initLogging(&logger, "Flash memory"); addConsoleAction("readconfig", readFromFlash); addConsoleAction("writeconfig", writeToFlash); addConsoleAction("resetconfig", doResetConfiguration); addConsoleActionP("rewriteconfig", (VoidPtr)rewriteConfig, engine); } #endif /* EFI_INTERNAL_FLASH */