/EEPROMWearLevel

Arduino EEPROMWearLevel reduces EEPROM wear by writing a new value to an other EEPROM location.

Primary LanguageC++Apache License 2.0Apache-2.0

Arduino EEPROMWearLevel Library

https://github.com/PRosenb/EEPROMWearLevel

EEPROMWearLevel bases on the EEPROM library included in the Arduino framework.
It reduces EEPROM wear by writing every new value to an other EEPROM location.

To do this, the current position needs to be stored at a known location.
A single bit on EEPROM can only be changed from 1 to 0. To change bits from 0 to 1, the whole byte has to be erased what changes all bits to 1. This erase process causes the main wear on the EEPROM.
EEPROMWearLevel uses control bytes to remember the current position when doing wear levelling and reduces wear of the control bytes by writing single bits from 1 to 0.

The begin() method must be called first and lets you define how many values need to be stored and how much of the EEPROM shall be used. Keep in mind that EEPROM_LAYOUT_VERSION always needs to be changed whenever you modify the EEPROM layout. Failure to do so will cause EEPROMWearLevel to malfunction and results in corruption of data.

Features

  • Same interface as EEPROM.h
  • Easy to use
  • Small footprint
  • Reduces EEPROM wear
  • Balances data writes inside of given area
  • Uses single bit writes to store current index
  • Can be used as ring buffer

Installation

Getting Started

Simple configuration:

#include <EEPROMWearLevel.h>

#define EEPROM_LAYOUT_VERSION 0
#define AMOUNT_OF_INDEXES 2

#define INDEX_CONFIGURATION_VAR1 0
#define INDEX_CONFIGURATION_VAR2 1

void setup() {
  Serial.begin(9600);
  while (!Serial);

  EEPROMwl.begin(EEPROM_LAYOUT_VERSION, AMOUNT_OF_INDEXES);

  writeConfiguration();
  readConfiguration();
}

void loop() {
}

void writeConfiguration() {
  // write a byte
  EEPROMwl.update(INDEX_CONFIGURATION_VAR1, 12);

  long var2 = 33333;
  EEPROMwl.put(INDEX_CONFIGURATION_VAR2, var2);
}

void readConfiguration() {
  byte var1 = EEPROMwl.read(INDEX_CONFIGURATION_VAR1);
  Serial.print(F("var1: "));
  Serial.println(var1);

  long var2 = -1;
  EEPROMwl.get(INDEX_CONFIGURATION_VAR2, var2);
  Serial.print(F("var2: "));
  Serial.println(var2);
}

Examples

The following example sketches are included in the EepromWearLevel library.
You can also see them in the Arduino Software (IDE) in menu File->Examples->EEPROMWearLevel.

Reference

Methods

/**
    Initialises EEPROMWearLevel. One of the begin() methods must be called
    before any other method.
    This method uses the whole EEPROM for wear levelling.
    @param layoutVersion your version of the EEPROM layout. When ever you change any value
    on the begin() method, the layoutVersion must be incremented. This will reset EEPROMWearLevel
    and initialise it and the EEPROM to the given values.
    @param amountOfIndexes the amount of indexes you want to use.
*/
void begin(const byte layoutVersion, const int amountOfIndexes);

/**
    Initialises EEPROMWearLevel. One of the begin() methods must be called
    before any other method.
    @param layoutVersion your version of the EEPROM layout. When ever you change any value
    on the begin() method, the layoutVersion must be incremented. This will reset EEPROMWearLevel
    and initialise it and the EEPROM to the given values.
    @param amountOfIndexes the amount of indexes you want to use.
    @param eepromLengthToUse the length of the EEPROM to use for wear levelling in case you want
    to use parts of the EEPROM for other purpose.
*/
void begin(const byte layoutVersion, const int amountOfIndexes, const int eepromLengthToUse);

/**
    Initialises EEPROMWearLevel. One of the begin() methods must be called
    before any other method.
    @param layoutVersion your version of the EEPROM layout. When ever you change any value
    on the begin() method, the layoutVersion must be incremented. This will reset EEPROMWearLevel
    and initialise it and the EEPROM to the given values.
    @param lengths array of the partition lengths to use on the EEPROM. The length includes the
    control bytes so the usable length for data is smaller. You can get the max length for data
    by calling getMaxDataLength().
    The array must contain amountOfIndexes entries.
    @param amountOfIndexes the amount of indexes you want to use.
*/
void begin(const byte layoutVersion, const int lengths[], const int amountOfIndexes);

/**
  Returns the amount of indexes that can be used. This value is defined by the begin() method.
*/
unsigned int length();

/**
   returns the maximum size a single element can be. If the element is larger than half of
   the maximum size, no wear levelling will be possible.
*/
int getMaxDataLength(const int idx) const;

/**
   reads one byte from idx
*/
uint8_t read(const int idx);

/**
   writes one byte if it is not the same as the last one.
*/
void update(const int idx, const uint8_t val);

/**
   writes one byte no matter what value was written before.
*/
void write(const int idx, const uint8_t val);

/**
   reads the last written value of idx or leaves t unchanged if no
   value written yet.
*/
template< typename T > T &get(const int idx, T &t);

/**
   writes a new value if it is not the same as the last one
*/
template< typename T > const T &put(const int idx, const T &t);

/**
   writes a new value no matter what value was written before.
*/
template< typename T > const T &putToNext(const int idx, const T &t);

/**
    returns the first index used to store data for this idx.
    This method can be called to use EEPROMWearLevel as a ring buffer.
*/
int getStartIndexEEPROM(const int idx);

/**
   returns the current read index of this idx.
    This method can be called to use EEPROMWearLevel as a ring buffer.
*/
int getCurrentIndexEEPROM(const int idx, int dataLength) ;

/**
   prints the EEPROMWearLevel status to print. Use Serial to
   print to the default serial port.
*/
void printStatus(Print &print);

/**
   prints content of the EEPROM to print
*/
void printBinary(Print &print, int startIndex, int endIndex);

Implementation Notes

Control Bytes

EEPROMWearLevel uses single bits to store the current index. Single bits of an EEPROM byte can only be programmed from 1 to 0. If a bit needs to change from 0 to 1, the whole byte must be cleared.
EEPROMWearLevel uses control bytes to store the location of the current index of the data. Every bit stands for one byte of data. A bit as 0 stands for data is in use, 1 for not yet used.

An empty control byte looks like this:

Index:              01234567
Bit representation: 11111111

Writing data of 2 bytes length changes it to (2 bits are programmed to 0 without clearing the whole byte):

Index:              01234567
Bit representation: 00111111

The control byte above states, that the first two indexes are used for data, the rest if free.
If you use larger partitions, the same is done with multiple control bytes. When all of them are marked as used (all bits 0), all bits in all control bytes of the partition are cleared what sets them back to 1.

EEPROM layout

EEPROMWearLevel first uses one byte to store the version. After that, the first partition starts. For every idx you use, one partition is allocated.
Assuming a configuration with a single partition of 18 bytes, it will be represented in EEPROM as follows:

EEPROM index: 0             1            2            3          4          .. 18
Description:  layoutVersion controlByte0 controlByte1 dataIndex0 dataIndex1 .. dataIndex15

In this example, two of the 18 bytes in the partition are used as control bytes while the other 16 is used to store the data.

The 'layoutVersion' is used to clear control bytes when their position on the EEPROM is changed by using other arguments on the method 'begin()'. It is therefore important to change the 'layoutVersion' whenever a change is made of the arguments of the 'begin()' method. A change of 'layoutVersion' causes EEPROMWearLevel to reset the required control bytes so that it can use them to store the indexes.

Contributions

Enhancements and improvements are welcome.

License

Arduino EEPROMWearLevel Library
Copyright (c) 2019 Peter Rosenberg (https://github.com/PRosenb).

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.