mirror of
https://github.com/0xFEEDC0DE64/arduino-esp32.git
synced 2025-07-05 06:46:31 +02:00
Converted EEPROM library to use nvs instead of partition. (#2678)
* Converted EEPROM library to use nvs instead of partition. Removed eeprom partition from all partition table CSV files. * Changed variable names, added some comments, formatting as per me-no-dev's requests * Checks for memory on malloc * Moved include nvs.h from header to code * Reworked the extra example to make it more clear how to actually use the library and persist data
This commit is contained in:
@ -1,10 +1,9 @@
|
||||
/*
|
||||
EEPROM.h -ported by Paolo Becchi to Esp32 from esp8266 EEPROM
|
||||
-Modified by Elochukwu Ifediora <ifedioraelochukwuc@gmail.com>
|
||||
-Converted to nvs lbernstone@gmail.com
|
||||
|
||||
Uses a one sector flash partition defined in partition table
|
||||
OR
|
||||
Multiple sector flash partitions defined by the name column in the partition table
|
||||
Uses a nvs byte array to emulate EEPROM
|
||||
|
||||
Copyright (c) 2014 Ivan Grokhotkov. All rights reserved.
|
||||
This file is part of the esp8266 core for Arduino environment.
|
||||
@ -25,72 +24,115 @@
|
||||
*/
|
||||
|
||||
#include "EEPROM.h"
|
||||
|
||||
#include <nvs.h>
|
||||
#include <esp_log.h>
|
||||
|
||||
EEPROMClass::EEPROMClass(uint32_t sector)
|
||||
: _sector(sector)
|
||||
, _data(0)
|
||||
EEPROMClass::EEPROMClass(void)
|
||||
: _data(0)
|
||||
, _size(0)
|
||||
, _dirty(false)
|
||||
, _mypart(NULL)
|
||||
, _handle(NULL)
|
||||
, _name("eeprom")
|
||||
, _user_defined_size(0)
|
||||
{
|
||||
}
|
||||
|
||||
EEPROMClass::EEPROMClass(uint32_t sector)
|
||||
// Only for compatiility, no sectors in nvs!
|
||||
: _data(0)
|
||||
, _size(0)
|
||||
, _dirty(false)
|
||||
, _handle(NULL)
|
||||
, _name("eeprom")
|
||||
, _user_defined_size(0)
|
||||
{
|
||||
}
|
||||
|
||||
EEPROMClass::EEPROMClass(const char* name, uint32_t user_defined_size)
|
||||
: _sector(0)
|
||||
, _data(0)
|
||||
: _data(0)
|
||||
, _size(0)
|
||||
, _dirty(false)
|
||||
, _mypart(NULL)
|
||||
, _handle(NULL)
|
||||
, _name(name)
|
||||
, _user_defined_size(user_defined_size)
|
||||
{
|
||||
}
|
||||
|
||||
EEPROMClass::EEPROMClass(void)
|
||||
: _sector(0)// (((uint32_t)&_SPIFFS_end - 0x40200000) / SPI_FLASH_SEC_SIZE))
|
||||
, _data(0)
|
||||
, _size(0)
|
||||
, _dirty(false)
|
||||
, _mypart(NULL)
|
||||
, _name("eeprom")
|
||||
, _user_defined_size(0)
|
||||
{
|
||||
}
|
||||
|
||||
EEPROMClass::~EEPROMClass() {
|
||||
// end();
|
||||
}
|
||||
|
||||
bool EEPROMClass::begin(size_t size) {
|
||||
if (size <= 0) {
|
||||
return false;
|
||||
if (!size) {
|
||||
return false;
|
||||
}
|
||||
if (size > SPI_FLASH_SEC_SIZE) {
|
||||
size = SPI_FLASH_SEC_SIZE;
|
||||
|
||||
esp_err_t res = nvs_open(_name, NVS_READWRITE, &_handle);
|
||||
if (res != ESP_OK) {
|
||||
log_e("Unable to open NVS namespace: %d", res);
|
||||
return false;
|
||||
}
|
||||
// _mypart = esp_partition_find_first(ESP_PARTITION_TYPE_DATA,ESP_PARTITION_SUBTYPE_ANY, EEPROM_FLASH_PARTITION_NAME);
|
||||
_mypart = esp_partition_find_first(ESP_PARTITION_TYPE_DATA, ESP_PARTITION_SUBTYPE_ANY, _name);
|
||||
if (_mypart == NULL) {
|
||||
return false;
|
||||
|
||||
size_t key_size = 0;
|
||||
res = nvs_get_blob(_handle, _name, NULL, &key_size);
|
||||
if(res != ESP_OK && res != ESP_ERR_NVS_NOT_FOUND) {
|
||||
log_e("Unable to read NVS key: %d", res);
|
||||
return false;
|
||||
}
|
||||
if (size < key_size) { // truncate
|
||||
log_w("truncating EEPROM from %d to %d", key_size, size);
|
||||
uint8_t* key_data = (uint8_t*) malloc(key_size);
|
||||
if(!key_data) {
|
||||
log_e("Not enough memory to truncate EEPROM!");
|
||||
return false;
|
||||
}
|
||||
nvs_get_blob(_handle, _name, key_data, &key_size);
|
||||
nvs_set_blob(_handle, _name, key_data, size);
|
||||
nvs_commit(_handle);
|
||||
free(key_data);
|
||||
}
|
||||
else if (size > key_size) { // expand or new
|
||||
size_t expand_size = size - key_size;
|
||||
uint8_t* expand_key = (uint8_t*) malloc(expand_size);
|
||||
if(!expand_key) {
|
||||
log_e("Not enough memory to expand EEPROM!");
|
||||
return false;
|
||||
}
|
||||
// check for adequate free space
|
||||
if(nvs_set_blob(_handle, "expand", expand_key, expand_size)) {
|
||||
log_e("Not enough space to expand EEPROM from %d to %d", key_size, size);
|
||||
free(expand_key);
|
||||
return false;
|
||||
}
|
||||
free(expand_key);
|
||||
nvs_erase_key(_handle, "expand");
|
||||
uint8_t* key_data = (uint8_t*) malloc(size);
|
||||
if(!key_data) {
|
||||
log_e("Not enough memory to expand EEPROM!");
|
||||
return false;
|
||||
}
|
||||
memset(key_data, 0, size);
|
||||
if(key_size) {
|
||||
log_i("Expanding EEPROM from %d to %d", key_size, size);
|
||||
// hold data while key is deleted
|
||||
nvs_get_blob(_handle, _name, key_data, &key_size);
|
||||
nvs_erase_key(_handle, _name);
|
||||
} else {
|
||||
log_i("New EEPROM of %d bytes", size);
|
||||
}
|
||||
nvs_commit(_handle);
|
||||
nvs_set_blob(_handle, _name, key_data, size);
|
||||
free(key_data);
|
||||
nvs_commit(_handle);
|
||||
}
|
||||
size = (size + 3) & (~3);
|
||||
|
||||
if (_data) {
|
||||
delete[] _data;
|
||||
}
|
||||
|
||||
_data = new uint8_t[size];
|
||||
_size = size;
|
||||
bool ret = false;
|
||||
if (esp_partition_read (_mypart, 0, (void *) _data, _size) == ESP_OK) {
|
||||
ret = true;
|
||||
}
|
||||
|
||||
return ret;
|
||||
nvs_get_blob(_handle, _name, _data, &_size);
|
||||
return true;
|
||||
}
|
||||
|
||||
void EEPROMClass::end() {
|
||||
@ -134,29 +176,21 @@ void EEPROMClass::write(int address, uint8_t value) {
|
||||
|
||||
bool EEPROMClass::commit() {
|
||||
bool ret = false;
|
||||
if (!_size)
|
||||
return false;
|
||||
if (!_dirty)
|
||||
return true;
|
||||
if (!_data)
|
||||
return false;
|
||||
|
||||
|
||||
if (esp_partition_erase_range(_mypart, 0, SPI_FLASH_SEC_SIZE) != ESP_OK)
|
||||
{
|
||||
log_e( "partition erase err.");
|
||||
if (!_size) {
|
||||
return false;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (esp_partition_write(_mypart, 0, (void *)_data, _size) == ESP_ERR_INVALID_SIZE)
|
||||
{
|
||||
log_e( "error in Write");
|
||||
}
|
||||
else
|
||||
{
|
||||
if (!_data) {
|
||||
return false;
|
||||
}
|
||||
if (!_dirty) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if (ESP_OK != nvs_set_blob(_handle, _name, _data, _size)) {
|
||||
log_e( "error in write");
|
||||
} else {
|
||||
_dirty = false;
|
||||
ret = true;
|
||||
}
|
||||
}
|
||||
|
||||
return ret;
|
||||
|
@ -1,10 +1,9 @@
|
||||
/*
|
||||
EEPROM.h -ported by Paolo Becchi to Esp32 from esp8266 EEPROM
|
||||
-Modified by Elochukwu Ifediora <ifedioraelochukwuc@gmail.com>
|
||||
-Converted to nvs lbernstone@gmail.com
|
||||
|
||||
Uses a one sector flash partition defined in partition table
|
||||
OR
|
||||
Multiple sector flash partitions defined by the name column in the partition table
|
||||
Uses a nvs byte array to emulate EEPROM
|
||||
|
||||
Copyright (c) 2014 Ivan Grokhotkov. All rights reserved.
|
||||
This file is part of the esp8266 core for Arduino environment.
|
||||
@ -30,19 +29,9 @@
|
||||
#define EEPROM_FLASH_PARTITION_NAME "eeprom"
|
||||
#endif
|
||||
#include <Arduino.h>
|
||||
extern "C" {
|
||||
|
||||
#include <stddef.h>
|
||||
#include <stdint.h>
|
||||
#include <string.h>
|
||||
#include <esp_partition.h>
|
||||
}
|
||||
typedef uint32_t nvs_handle;
|
||||
|
||||
//
|
||||
// need to define AT LEAST a flash partition for EEPROM with above name
|
||||
//
|
||||
// eeprom , data , 0x99, start address, 0x1000
|
||||
//
|
||||
class EEPROMClass {
|
||||
public:
|
||||
EEPROMClass(uint32_t sector);
|
||||
@ -117,11 +106,10 @@ class EEPROMClass {
|
||||
template <class T> T writeAll (int address, const T &);
|
||||
|
||||
protected:
|
||||
uint32_t _sector;
|
||||
nvs_handle _handle;
|
||||
uint8_t* _data;
|
||||
size_t _size;
|
||||
bool _dirty;
|
||||
const esp_partition_t * _mypart;
|
||||
const char* _name;
|
||||
uint32_t _user_defined_size;
|
||||
};
|
||||
|
Reference in New Issue
Block a user