Some C++ macros to simplify creating a block of EEPROM data
This set of macros simplifies creating a block of data in EEPROM. It allows you to create read, write, set, get, and print methods for each variable in a block.
Usage: In the appropriate places (see below), you will write
#define EEPROM_something
#include "EEPROMstate.h"
To do this right, you have to set up a header file which will define all your variables. This file is normally called EEPROMstate.h, and a sample is supplied with this source. Its general form is shown below. Note that there are no "include guards" because this file is included many times in your build, sometimes multiple times within a single module. By doing different #defines before each include of this file, it will generate different kinds of code.
// Note that there are no "include guards" because this file can be included several times
//#included "EEPROMstate.h"
#define EEPROM_DEFINE
#include "Access.h"
// Use the EEPROMstate and EEPROMstate2 macros to define the fields
// of your EEPROM data block
EEPROMstate2(offset, uint32_t, Signature, HEX, thisSignature);
EEPROMstate2(offset, uint16_t, Version, HEX, thisVersion);
EEPROMstate(offset, bool, Debugging, false);
// end of user-definable fields
#define EEPROM_CLEAR
#include "access.h"
In this example, there are three fields defined. The first field is the "signature", a 32-bit value which identifies this block as a valid block of data. You will typically set this to a specific value you have chosen, and when you read in the block, the first thing you do is compare this 32-bit value with the known 32-bit value. If they are the same, then the rest of the data is valid and can be trusted. If they are not the same, it means that the EEPROM has never been written to by this program and you should not turst any of the fields to be valid. So at this point, you want to call a function that sets the values to default values, including the signature and version values.
In the above examples, "thisSignature" can be any 32-bit value, as long as the value is either a literal constant (e.g., 0xDEADBEEF) or a name that is defined at the point where it can be set. More on this below.
The version number is your own version number that tells you what "version" of the code you are using. For example, the version number might first be 0x0001. Suppose that you add a new field (always add at the endf of the list). You might then choose to indicate what has been written by changing the version number to 0x0002. It is up to you to write code that knows that if the version number is 1, version 2 values are not valid. Tedious, but I have not come up with a better method.
Note also that you will not have to define these variables "by hand". The EEPROMstate/EEPROMstate2 macros handle this for you.
Use the EEPROMstate or EEPROMstate2 macros to list the variables. For these, the crucial parameter is the type and the name parameters:
EEPROMstate (offset, type, name, value)
EEPROMstate2(offset, type, name, format, value)
The usage is as follows:
class Whatever {
class MyEEPROMdata {
#define EEPROM_VARIABLE_DECLARATION
#include "EEPROMstate.h"
} eeprom; // class MyEEPROMPdata
...
}; // class Whatever
I have assumed that the collection of EEPROM data is going to be a member of the Whatever class. I do hardwire the name eeprom into some of the macros, so that name is required.
By defining EEPROM_VARIABLE_DECLARATION, the result of this becomes
class Whatever { class MyEEPROMdata { protected: uint32_t Signature; protected: uint16_t Version; protected: bool Debugging; } eeprom; // class MyEEPROMdata
Note that for the visibility of the variables, protected: is used. Therefore, there is no way to set or examine these values. We have to implement setter and getter methods for this class. This is done by defining the symbols EEPROM_SET_DECLARATION and EEPROM_GET_DECLARATION:
class MyEEPROMdata {
#define EEPROM_VARIABLE_DECLARATION
// from EEPROM_VARIABLE_DECLARATION
#include "EEPROMstate.h"
#define EEPROM_SET_DECLARATION
// from EEPROM_SET_DECLARATION
#include "EEPROMstate.h"
// from EEPROM_GET_DECLARATION
#define EEPROM_GET_DECLARATION
#include "EEPROMstate.h"
} eeprom; // class MyEEPROMPdata
Ater macro expansions, the class looks like this:
class MyEEPROMdata {
// from EEPROM_VARIABLE_DECLARATION
protected: uint32_t Signature;
protected: uint16_t Version;
protected: bool Debugging;
// from EEPROM_SET_DECLARATION
public: inline void setSignature(uint32_t val) { Signature = val; }
public: inline void setVersion(uint16_t val) { Version = val; }
public: inline void setDebugging(bool val) { Debugging = val; }
// from EEPROM_GET_DECLARATION
public: inline uint32_t getSignature() { return Signature; }
public: inline uint16_t getVersion() { return Version; }
public: inline bool getDebugging() { return Debugging; }
} eeprom; // class MyEEPROMdata
For each value, there is a print method associated with it. In this case, the name and mode parameters are used. For EEPROMstate, the only formatting option is the default used by the print/println methods of the output device. The programmer has to define some inline methods to make this work. The macros expand to calls on ConsolePrint, so the meaning depends on what ConsolePrint is defined as. Because there are variants of these with 0, 1 or 2 parameters, they cannot directly be done as macros for anything but the Serial device (which already has them defined). For example, for simple output via Serial.print, before invoking the EEPROMstate and EEPROMstate2 macros, a simple #define can be used:
#define ConsolePrint Serial.print
#define ConsolePrintln Serial.println
But for more sophisticated output, such as writing to a local TFT screen, it might be necessary to do something more complex:
#dfine ConsolePrint TFTprint
public: template <typename T> inline void TFTprint(T x) { TFT.print(x); }
public: template <typename T> inline void TFTprint(T x, int mode) { TFT.print(x, mode); }
public: template <typename T> inline void TFTprintln(T x) { TFT.println(x); }
public: template <typename T> inline void TFTprintln(T x, int mode) { TFT.println(x, mode); }
public: inline void TFTprintln() { TFT.println(); }
More commonly, the TFT variable will be a reference to an object which already has these methods defined, in which case the simpler solution
#define ConsolPrint TFT.print
#define ConsolePrintln TFT.println
The effect of defining EEPROM_PRINT_DECLARATION is to declare the print routines, for example
class MyEEPROMdata {
... as above
#define ConsolePrint Serial.print
#defone ConsolePrintln Serial.println
#define EEPROM_PRINT_DECLARATION
// from EEPROM_PRINT_DECLARATION
#include "EEPROMstate.h"
is to generate the code
// from EEPROM_PRINT_DECLARATION
public: void printSignature()
{
Serial.print(F("Signature"));
Serial.print(F(": "));
Serial.print(getSignature(), HEX);
Serial.println();
}
public: void PrintVersion()
{
Serial.print(F("Version"));
Serial.print(F(": "));
Serial.print(getVersion(), HEX);
Serial.println();
}
public: void PrintDebuggin()
{
Serial.print(F("Debugging"));
Serial.print(F(": "));
Serial.print(getDebugging());
Serial.println();
}
It is often convenient to have a method that simply prints out, in some nice format, all the values. This can be handled using the EEPROM_PRINT_EXECUTE conditional:
class MyEEPROMdata {
... as above
public: void print() {
#define EEPROM_PRINT_EXECUTE
#include "EEPROMstate.h"
} // MyEEPROMdata::print
...
}
Which, for our example, generates:
class MyEEPROMdata {
... as previous
public: void print() {
printSignature();
printVersion();
printDebugging();
} // MyEEPROMdata::print
...
} eeprom; // class MyEEPROMdata
### Summary of **EEPROMstate**/**EEPROMstate2** parameters
Parameter | Meaning
--------- | -------
_offset_ | _reserved for future use_
_type_ | The type of the value (declarations, setters & getters)
_name_ | The name of the variable (declarations, setters & getters, debug printing
_mode_ | (**EEPROMstate2** *only*) The formatting mode. For integer values, this can be **DEC**, **OCT** or **HEX**, for **float** and **double** values, it is the number of decimal places.
_value_ | The value to be used as the default value (initialization).