rusefi/firmware/controllers/flash_main.cpp

207 lines
6.4 KiB
C++
Raw Normal View History

2015-07-10 06:01:56 -07:00
/**
* @file flash_main.cpp
* @brief Higher-level logic of saving data into internal flash memory
*
*
* @date Sep 19, 2013
2018-01-20 17:55:31 -08:00
* @author Andrey Belomutskiy, (c) 2012-2018
2015-07-10 06:01:56 -07:00
*/
2018-09-16 19:39:46 -07:00
#include "global.h"
2019-04-12 19:07:03 -07:00
#if EFI_INTERNAL_FLASH
2019-07-05 17:03:32 -07:00
#include "os_access.h"
2015-07-10 06:01:56 -07:00
#include "flash_main.h"
#include "eficonsole.h"
2015-07-10 06:01:56 -07:00
#include "flash.h"
2018-02-03 07:55:15 -08:00
#include "engine_math.h"
2015-07-10 06:01:56 -07:00
2015-07-13 17:02:18 -07:00
// this message is part of console API, see FLASH_SUCCESS_MSG in java code
#define FLASH_SUCCESS_MSG "FLASH_SUCESS"
2019-04-12 19:07:03 -07:00
#if EFI_TUNER_STUDIO
2015-07-10 06:01:56 -07:00
#include "tunerstudio.h"
#endif
#include "engine_controller.h"
static bool needToWriteConfiguration = false;
EXTERN_ENGINE;
static Logging* logger;
extern persistent_config_container_s persistentState;
extern engine_configuration_s *engineConfiguration;
2017-03-19 16:29:54 -07:00
/**
* this address needs to be above 'flash' region available for firmware
* todo: an ideal solution would be to define this address in the .ld / .icf mapping file
*/
2019-01-27 14:53:11 -08:00
#ifndef FLASH_ADDR
2017-03-19 16:29:54 -07:00
#define FLASH_ADDR 0x080E0000
2019-01-27 14:53:11 -08:00
#endif
2015-07-10 06:01:56 -07:00
#define PERSISTENT_SIZE sizeof(persistent_config_container_s)
2017-02-13 22:03:01 -08:00
/**
* https://sourceforge.net/p/rusefi/tickets/335/
*
* In order to preserve at least one copy of the tune in case of electrical issues address of second configuration copy
* should be in a different sector of flash since complete flash sectors are erased on write.
2017-02-13 22:03:01 -08:00
*/
#ifndef FLASH_ADDR_SECOND_COPY
#define FLASH_ADDR_SECOND_COPY 0x080C0000
#endif
2017-02-13 22:03:01 -08:00
2015-07-10 06:01:56 -07:00
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");
writeToFlashNow();
}
void writeToFlashNow(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);
2015-07-13 17:02:18 -07:00
crc_t crcResult = flashStateCrc(&persistentState);
persistentState.value = crcResult;
2015-07-10 06:01:56 -07:00
scheduleMsg(logger, "Reseting flash: size=%d", PERSISTENT_SIZE);
flashErase(FLASH_ADDR, PERSISTENT_SIZE);
2015-07-13 17:02:18 -07:00
scheduleMsg(logger, "Flashing with CRC=%d", crcResult);
2015-07-10 06:01:56 -07:00
efitimems_t nowMs = currentTimeMillis();
2015-07-13 17:02:18 -07:00
int result = flashWrite(FLASH_ADDR, (const char *) &persistentState, PERSISTENT_SIZE);
flashErase(FLASH_ADDR_SECOND_COPY, PERSISTENT_SIZE);
2017-02-13 22:03:01 -08:00
flashWrite(FLASH_ADDR_SECOND_COPY, (const char *) &persistentState, PERSISTENT_SIZE);
2015-07-13 17:02:18 -07:00
scheduleMsg(logger, "Flash programmed in %dms", currentTimeMillis() - nowMs);
2016-01-11 14:01:33 -08:00
bool isSuccess = result == FLASH_RETURN_SUCCESS;
2015-07-13 17:02:18 -07:00
if (isSuccess) {
scheduleMsg(logger, FLASH_SUCCESS_MSG);
} else {
scheduleMsg(logger, "Flashing failed");
}
2018-02-03 07:55:15 -08:00
assertEngineReference();
2017-05-19 18:52:10 -07:00
resetMaxValues();
2015-07-10 06:01:56 -07:00
}
static bool isValidCrc(persistent_config_container_s *state) {
crc_t result = flashStateCrc(state);
int isValidCrc_b = result == state->value;
return isValidCrc_b;
}
static void doResetConfiguration(void) {
2017-05-15 20:28:49 -07:00
resetConfigurationExt(logger, engineConfiguration->engineType PASS_ENGINE_PARAMETER_SUFFIX);
2015-07-10 06:01:56 -07:00
}
2015-12-21 20:01:27 -08:00
persisted_configuration_state_e flashState;
2015-07-10 06:01:56 -07:00
2017-02-13 22:03:01 -08:00
static persisted_configuration_state_e doReadConfiguration(flashaddr_t address, Logging * logger) {
printMsg(logger, "readFromFlash %x", address);
flashRead(address, (char *) &persistentState, PERSISTENT_SIZE);
if (!isValidCrc(&persistentState)) {
return CRC_FAILED;
} else if (persistentState.version != FLASH_DATA_VERSION || persistentState.size != PERSISTENT_SIZE) {
return INCOMPATIBLE_VERSION;
} else {
return PC_OK;
}
}
2016-04-03 11:02:00 -07:00
/**
* this method could and should be executed before we have any
* connectivity so no console output here
*/
2016-04-03 15:02:39 -07:00
persisted_configuration_state_e readConfiguration(Logging * logger) {
efiAssert(CUSTOM_ERR_ASSERT, getCurrentRemainingStack() > EXPECTED_REMAINING_STACK, "read f", PC_ERROR);
2017-02-13 22:03:01 -08:00
persisted_configuration_state_e result = doReadConfiguration(FLASH_ADDR, logger);
if (result != PC_OK) {
printMsg(logger, "Reading second configuration copy");
result = doReadConfiguration(FLASH_ADDR_SECOND_COPY, logger);
}
if (result == CRC_FAILED) {
2016-12-13 11:01:43 -08:00
warning(CUSTOM_ERR_FLASH_CRC_FAILED, "flash CRC failed");
2017-05-15 20:28:49 -07:00
resetConfigurationExt(logger, DEFAULT_ENGINE_TYPE PASS_ENGINE_PARAMETER_SUFFIX);
2017-02-13 22:03:01 -08:00
} else if (result == INCOMPATIBLE_VERSION) {
2017-05-15 20:28:49 -07:00
resetConfigurationExt(logger, engineConfiguration->engineType PASS_ENGINE_PARAMETER_SUFFIX);
2015-07-10 06:01:56 -07:00
} else {
2015-12-21 20:01:27 -08:00
/**
* At this point we know that CRC and version number is what we expect. Safe to assume it's a valid configuration.
*/
2017-05-15 20:28:49 -07:00
applyNonPersistentConfiguration(logger PASS_ENGINE_PARAMETER_SUFFIX);
2015-07-10 06:01:56 -07:00
}
// we can only change the state after the CRC check
2016-12-25 08:02:42 -08:00
engineConfiguration->byFirmwareVersion = getRusEfiVersion();
2017-08-31 04:53:41 -07:00
validateConfiguration(PASS_ENGINE_PARAMETER_SIGNATURE);
2016-04-03 11:02:00 -07:00
return result;
}
void readFromFlash(void) {
2016-04-03 15:02:39 -07:00
persisted_configuration_state_e result = readConfiguration(logger);
2015-12-21 20:01:27 -08:00
if (result == CRC_FAILED) {
printMsg(logger, "Need to reset flash to default due to CRC");
} else if (result == INCOMPATIBLE_VERSION) {
printMsg(logger, "Resetting but saving engine type [%d]", engineConfiguration->engineType);
} else {
printMsg(logger, "Got valid configuration from flash!");
}
2015-07-10 06:01:56 -07:00
}
static void rewriteConfig(void) {
doResetConfiguration();
writeToFlashNow();
}
static void writeConfigCommand() {
#if EFI_TUNER_STUDIO
// on start-up rusEfi would read from working copy of TS while
// we have a lot of console commands which write into real copy of configuration directly
// we have a bit of a mess here
syncTunerStudioCopy();
#endif /* EFI_TUNER_STUDIO */
writeToFlashNow();
}
2015-07-10 06:01:56 -07:00
void initFlash(Logging *sharedLogger) {
logger = sharedLogger;
addConsoleAction("readconfig", readFromFlash);
/**
* This would write NOW (you should not be doing this while connected to real engine)
*/
addConsoleAction(CMD_WRITECONFIG, writeConfigCommand);
2019-04-12 19:07:03 -07:00
#if EFI_TUNER_STUDIO
2015-07-10 06:01:56 -07:00
/**
* This would schedule write to flash once the engine is stopped
*/
addConsoleAction("burnconfig", requestBurn);
#endif
addConsoleAction("resetconfig", doResetConfiguration);
addConsoleAction("rewriteconfig", rewriteConfig);
}
#endif /* EFI_INTERNAL_FLASH */