2018-01-11 22:29:08 +01:00
|
|
|
/**
|
|
|
|
* Marlin 3D Printer Firmware
|
2020-02-03 08:00:57 -06:00
|
|
|
* Copyright (c) 2020 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
|
2018-01-11 22:29:08 +01:00
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
2020-07-23 05:20:14 +02:00
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
2018-01-11 22:29:08 +01:00
|
|
|
*
|
|
|
|
*/
|
2019-07-09 23:54:34 -05:00
|
|
|
#if defined(STM32GENERIC) && (defined(STM32F4) || defined(STM32F7))
|
2018-01-15 02:28:39 -06:00
|
|
|
|
2018-01-11 22:29:08 +01:00
|
|
|
#include "../../inc/MarlinConfig.h"
|
|
|
|
|
2020-03-11 16:41:18 -05:00
|
|
|
#if ENABLED(FLASH_EEPROM_EMULATION)
|
2019-07-09 23:54:34 -05:00
|
|
|
|
2020-04-29 16:31:58 -05:00
|
|
|
#include "../shared/eeprom_api.h"
|
2019-07-09 23:54:34 -05:00
|
|
|
#include "eeprom_emul.h"
|
2018-01-11 22:29:08 +01:00
|
|
|
|
|
|
|
// FLASH_FLAG_PGSERR (Programming Sequence Error) was renamed to
|
2019-07-09 23:54:34 -05:00
|
|
|
// FLASH_FLAG_ERSERR (Erasing Sequence Error) in STM32F4/7
|
|
|
|
|
|
|
|
#ifdef STM32F7
|
|
|
|
#define FLASH_FLAG_PGSERR FLASH_FLAG_ERSERR
|
|
|
|
#else
|
|
|
|
//#define FLASH_FLAG_PGSERR FLASH_FLAG_ERSERR
|
|
|
|
#endif
|
2018-01-11 22:29:08 +01:00
|
|
|
|
2020-04-29 16:31:58 -05:00
|
|
|
void ee_write_byte(uint8_t *pos, unsigned char value) {
|
2018-01-11 22:29:08 +01:00
|
|
|
HAL_FLASH_Unlock();
|
|
|
|
__HAL_FLASH_CLEAR_FLAG(FLASH_FLAG_EOP | FLASH_FLAG_OPERR | FLASH_FLAG_WRPERR |FLASH_FLAG_PGAERR | FLASH_FLAG_PGPERR | FLASH_FLAG_PGSERR);
|
2018-02-25 00:13:46 -06:00
|
|
|
|
2020-04-28 02:27:55 -05:00
|
|
|
const unsigned eeprom_address = (unsigned)pos;
|
2019-07-09 22:49:58 -05:00
|
|
|
if (EE_WriteVariable(eeprom_address, uint16_t(value)) != EE_OK)
|
|
|
|
for (;;) HAL_Delay(1); // Spin forever until watchdog reset
|
2018-02-25 00:13:46 -06:00
|
|
|
|
2018-01-11 22:29:08 +01:00
|
|
|
HAL_FLASH_Lock();
|
|
|
|
}
|
|
|
|
|
2020-04-29 16:31:58 -05:00
|
|
|
uint8_t ee_read_byte(uint8_t *pos) {
|
2019-07-09 22:49:58 -05:00
|
|
|
uint16_t data = 0xFF;
|
2020-04-28 02:27:55 -05:00
|
|
|
const unsigned eeprom_address = (unsigned)pos;
|
2019-07-09 22:49:58 -05:00
|
|
|
(void)EE_ReadVariable(eeprom_address, &data); // Data unchanged on error
|
|
|
|
return uint8_t(data);
|
2018-01-11 22:29:08 +01:00
|
|
|
}
|
|
|
|
|
2020-05-22 02:15:40 -05:00
|
|
|
#ifndef MARLIN_EEPROM_SIZE
|
|
|
|
#error "MARLIN_EEPROM_SIZE is required for Flash-based EEPROM."
|
|
|
|
#endif
|
|
|
|
size_t PersistentStore::capacity() { return MARLIN_EEPROM_SIZE; }
|
|
|
|
|
2020-04-29 16:31:58 -05:00
|
|
|
bool PersistentStore::access_finish() { return true; }
|
|
|
|
|
2020-05-22 02:15:40 -05:00
|
|
|
bool PersistentStore::access_start() {
|
2020-05-01 00:04:55 -05:00
|
|
|
static bool ee_initialized = false;
|
|
|
|
if (!ee_initialized) {
|
|
|
|
HAL_FLASH_Unlock();
|
|
|
|
|
|
|
|
__HAL_FLASH_CLEAR_FLAG(FLASH_FLAG_EOP | FLASH_FLAG_OPERR | FLASH_FLAG_WRPERR |FLASH_FLAG_PGAERR | FLASH_FLAG_PGPERR | FLASH_FLAG_PGSERR);
|
|
|
|
|
|
|
|
/* EEPROM Init */
|
|
|
|
if (EE_Initialize() != EE_OK)
|
|
|
|
for (;;) HAL_Delay(1); // Spin forever until watchdog reset
|
|
|
|
|
|
|
|
HAL_FLASH_Lock();
|
|
|
|
ee_initialized = true;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-04-29 16:31:58 -05:00
|
|
|
bool PersistentStore::write_data(int &pos, const uint8_t *value, size_t size, uint16_t *crc) {
|
|
|
|
while (size--) {
|
|
|
|
uint8_t * const p = (uint8_t * const)pos;
|
|
|
|
uint8_t v = *value;
|
|
|
|
// EEPROM has only ~100,000 write cycles,
|
|
|
|
// so only write bytes that have changed!
|
|
|
|
if (v != ee_read_byte(p)) {
|
|
|
|
ee_write_byte(p, v);
|
|
|
|
if (ee_read_byte(p) != v) {
|
|
|
|
SERIAL_ECHO_MSG(STR_ERR_EEPROM_WRITE);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
crc16(crc, &v, 1);
|
|
|
|
pos++;
|
|
|
|
value++;
|
|
|
|
};
|
|
|
|
return false;
|
2018-01-11 22:29:08 +01:00
|
|
|
}
|
|
|
|
|
2020-04-29 16:31:58 -05:00
|
|
|
bool PersistentStore::read_data(int &pos, uint8_t* value, size_t size, uint16_t *crc, const bool writing/*=true*/) {
|
|
|
|
do {
|
|
|
|
uint8_t c = ee_read_byte((uint8_t*)pos);
|
|
|
|
if (writing) *value = c;
|
|
|
|
crc16(crc, &c, 1);
|
|
|
|
pos++;
|
|
|
|
value++;
|
|
|
|
} while (--size);
|
|
|
|
return false;
|
2018-01-11 22:29:08 +01:00
|
|
|
}
|
|
|
|
|
2020-03-11 16:41:18 -05:00
|
|
|
#endif // FLASH_EEPROM_EMULATION
|
2019-07-09 23:54:34 -05:00
|
|
|
#endif // STM32GENERIC && (STM32F4 || STM32F7)
|