mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 09:52:45 +01:00
499 lines
12 KiB
C
499 lines
12 KiB
C
/*
|
|
* Copyright (C) 2018 Acutam Automation, LLC
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
* directory for more details.
|
|
*/
|
|
|
|
/**
|
|
* @ingroup sys_eepreg
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief eepreg implementation
|
|
*
|
|
* @author Matthew Blue <matthew.blue.neuro@gmail.com>
|
|
* @}
|
|
*/
|
|
|
|
#include <errno.h>
|
|
#include <limits.h>
|
|
#include <stdint.h>
|
|
#include <string.h>
|
|
|
|
#include "eepreg.h"
|
|
#include "periph/eeprom.h"
|
|
|
|
#define ENABLE_DEBUG 0
|
|
#include "debug.h"
|
|
|
|
/* EEPREG magic number */
|
|
static const char eepreg_magic[] = "RIOTREG";
|
|
|
|
/* constant lengths */
|
|
#define MAGIC_SIZE (sizeof(eepreg_magic) - 1) /* -1 to remove null */
|
|
#define ENT_LEN_SIZ (1U)
|
|
|
|
/* constant locations */
|
|
#define REG_START (EEPROM_RESERV_CPU_LOW + EEPROM_RESERV_BOARD_LOW)
|
|
#define REG_MAGIC_LOC (REG_START)
|
|
#define REG_END_PTR_LOC (REG_MAGIC_LOC + MAGIC_SIZE)
|
|
#define REG_ENT1_LOC (REG_END_PTR_LOC + EEPREG_PTR_LEN)
|
|
#define DAT_START (EEPROM_SIZE - EEPROM_RESERV_CPU_HI \
|
|
- EEPROM_RESERV_BOARD_HI - 1)
|
|
|
|
static inline uint32_t _read_meta_uint(uint32_t loc)
|
|
{
|
|
uint8_t data[4];
|
|
uint32_t ret;
|
|
|
|
eeprom_read(loc, data, EEPREG_PTR_LEN);
|
|
|
|
/* unused array members will be discarded */
|
|
ret = ((uint32_t)data[0] << 24)
|
|
| ((uint32_t)data[1] << 16)
|
|
| ((uint32_t)data[2] << 8)
|
|
| ((uint32_t)data[3]);
|
|
|
|
/* bit shift to discard unused array members */
|
|
ret >>= 8 * (4 - EEPREG_PTR_LEN);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static inline void _write_meta_uint(uint32_t loc, uint32_t val)
|
|
{
|
|
uint8_t data[4];
|
|
|
|
val <<= 8 * (4 - EEPREG_PTR_LEN);
|
|
|
|
data[0] = (uint8_t)(val >> 24);
|
|
data[1] = (uint8_t)(val >> 16);
|
|
data[2] = (uint8_t)(val >> 8);
|
|
data[3] = (uint8_t)val;
|
|
|
|
eeprom_write(loc, data, EEPREG_PTR_LEN);
|
|
}
|
|
|
|
static inline uint32_t _get_reg_end(void)
|
|
{
|
|
return _read_meta_uint(REG_END_PTR_LOC);
|
|
}
|
|
|
|
static inline void _set_reg_end(uint32_t loc)
|
|
{
|
|
_write_meta_uint(REG_END_PTR_LOC, loc);
|
|
}
|
|
|
|
static inline uint32_t _get_last_loc(uint32_t reg_end)
|
|
{
|
|
if (reg_end == REG_ENT1_LOC) {
|
|
/* no entries yet */
|
|
return DAT_START;
|
|
}
|
|
|
|
return _read_meta_uint(reg_end - EEPREG_PTR_LEN);
|
|
}
|
|
|
|
static inline uint32_t _calc_free_space(uint32_t reg_end, uint32_t last_loc)
|
|
{
|
|
return last_loc - reg_end;
|
|
}
|
|
|
|
static inline uint8_t _get_meta_len(uint32_t meta_loc)
|
|
{
|
|
return eeprom_read_byte(meta_loc);
|
|
}
|
|
|
|
static inline void _set_meta_len(uint32_t meta_loc, uint8_t meta_len)
|
|
{
|
|
eeprom_write_byte(meta_loc, meta_len);
|
|
}
|
|
|
|
static inline uint32_t _get_data_loc(uint32_t meta_loc, uint8_t meta_len)
|
|
{
|
|
/* data location is at the end of meta-data */
|
|
return _read_meta_uint(meta_loc + meta_len - EEPREG_PTR_LEN);
|
|
}
|
|
|
|
static inline void _set_data_loc(uint32_t meta_loc, uint8_t meta_len,
|
|
uint32_t data_loc)
|
|
{
|
|
/* data location is at the end of meta-data */
|
|
_write_meta_uint(meta_loc + meta_len - EEPREG_PTR_LEN, data_loc);
|
|
}
|
|
|
|
static inline uint8_t _calc_name_len(uint8_t meta_len)
|
|
{
|
|
/* entry contents: meta-data length, name, data pointer */
|
|
return meta_len - ENT_LEN_SIZ - EEPREG_PTR_LEN;
|
|
}
|
|
|
|
static inline void _get_name(uint32_t meta_loc, char *name, uint8_t meta_len)
|
|
{
|
|
/* name is after entry length */
|
|
eeprom_read(meta_loc + ENT_LEN_SIZ, (uint8_t *)name,
|
|
_calc_name_len(meta_len));
|
|
}
|
|
|
|
static inline int _cmp_name(uint32_t meta_loc, const char *name,
|
|
uint8_t meta_len)
|
|
{
|
|
/* name is after entry length */
|
|
uint32_t loc = meta_loc + ENT_LEN_SIZ;
|
|
|
|
uint8_t len = _calc_name_len(meta_len);
|
|
|
|
uint8_t offset;
|
|
for (offset = 0; offset < len; offset++) {
|
|
if (name[offset] == '\0') {
|
|
/* entry name is longer than name */
|
|
return 0;
|
|
}
|
|
|
|
if (eeprom_read_byte(loc + offset) != (uint8_t)name[offset]) {
|
|
/* non-matching character */
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
if (name[offset] == '\0') {
|
|
/* entry name is the same length as name */
|
|
return 1;
|
|
}
|
|
|
|
/* entry name is shorter than name */
|
|
return 0;
|
|
}
|
|
|
|
static inline uint32_t _get_meta_loc(const char *name)
|
|
{
|
|
uint32_t meta_loc = REG_ENT1_LOC;
|
|
uint32_t reg_end = _get_reg_end();
|
|
|
|
while (meta_loc < reg_end) {
|
|
uint8_t meta_len = _get_meta_len(meta_loc);
|
|
|
|
if (_cmp_name(meta_loc, name, meta_len)) {
|
|
return meta_loc;
|
|
}
|
|
|
|
meta_loc += meta_len;
|
|
}
|
|
|
|
/* no meta-data found */
|
|
return (uint32_t)UINT_MAX;
|
|
}
|
|
|
|
static inline uint32_t _get_data_len(uint32_t meta_loc, uint32_t data_loc)
|
|
{
|
|
uint32_t prev_loc;
|
|
if (meta_loc == REG_ENT1_LOC) {
|
|
prev_loc = DAT_START;
|
|
}
|
|
else {
|
|
/* previous entry data pointer is just before this entry */
|
|
prev_loc = _read_meta_uint(meta_loc - EEPREG_PTR_LEN);
|
|
}
|
|
|
|
return prev_loc - data_loc;
|
|
}
|
|
|
|
static inline int _new_entry(const char *name, uint32_t data_len)
|
|
{
|
|
uint32_t reg_end = _get_reg_end();
|
|
uint32_t last_loc = _get_last_loc(reg_end);
|
|
uint32_t free_space = _calc_free_space(reg_end, last_loc);
|
|
|
|
uint8_t name_len = (uint8_t)strlen(name);
|
|
uint8_t meta_len = ENT_LEN_SIZ + name_len + EEPREG_PTR_LEN;
|
|
|
|
/* check to see if there is enough room */
|
|
if (free_space < meta_len + data_len) {
|
|
return -ENOSPC;
|
|
}
|
|
|
|
/* set the length of the meta-data */
|
|
_set_meta_len(reg_end, meta_len);
|
|
|
|
/* write name of entry */
|
|
eeprom_write(reg_end + ENT_LEN_SIZ, (uint8_t *)name, name_len);
|
|
|
|
/* set the location of the data */
|
|
_set_data_loc(reg_end, meta_len, last_loc - data_len);
|
|
|
|
/* update end of the registry */
|
|
_set_reg_end(reg_end + meta_len);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static inline void _move_data(uint32_t oldpos, uint32_t newpos, uint32_t len)
|
|
{
|
|
for (uint32_t count = 0; count < len; count++) {
|
|
uint32_t offset;
|
|
|
|
if (newpos < oldpos) {
|
|
/* move from beginning of data */
|
|
offset = count;
|
|
}
|
|
else {
|
|
/* move from end of data */
|
|
offset = len - count;
|
|
}
|
|
|
|
uint8_t byte = eeprom_read_byte(oldpos + offset);
|
|
|
|
eeprom_write_byte(newpos + offset, byte);
|
|
}
|
|
}
|
|
|
|
int eepreg_add(uint32_t *pos, const char *name, uint32_t len)
|
|
{
|
|
int ret = eepreg_check();
|
|
if (ret == -ENOENT) {
|
|
/* reg does not exist, so make a new one */
|
|
eepreg_reset();
|
|
}
|
|
else if (ret < 0) {
|
|
DEBUG("[eepreg_add] eepreg_check failed\n");
|
|
return ret;
|
|
}
|
|
|
|
uint32_t reg_end = _get_reg_end();
|
|
|
|
uint32_t meta_loc = _get_meta_loc(name);
|
|
|
|
if (meta_loc == (uint32_t)UINT_MAX) {
|
|
/* entry does not exist, so make a new one */
|
|
|
|
/* location of the new data */
|
|
*pos = _get_last_loc(reg_end) - len;
|
|
|
|
if (_new_entry(name, len) < 0) {
|
|
DEBUG("[eepreg_add] not enough space for %s\n", name);
|
|
return -ENOSPC;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
*pos = _get_data_loc(meta_loc, _get_meta_len(meta_loc));
|
|
|
|
if (len != _get_data_len(meta_loc, *pos)) {
|
|
DEBUG("[eepreg_add] %s already exists with different length\n", name);
|
|
return -EADDRINUSE;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int eepreg_read(uint32_t *pos, const char *name)
|
|
{
|
|
int ret = eepreg_check();
|
|
if (ret < 0) {
|
|
DEBUG("[eepreg_read] eepreg_check failed\n");
|
|
return ret;
|
|
}
|
|
|
|
uint32_t meta_loc = _get_meta_loc(name);
|
|
|
|
if (meta_loc == (uint32_t)UINT_MAX) {
|
|
DEBUG("[eepreg_read] no entry for %s\n", name);
|
|
return -ENOENT;
|
|
}
|
|
|
|
*pos = _get_data_loc(meta_loc, _get_meta_len(meta_loc));
|
|
|
|
return 0;
|
|
}
|
|
|
|
int eepreg_write(uint32_t *pos, const char *name, uint32_t len)
|
|
{
|
|
uint32_t reg_end = _get_reg_end();
|
|
|
|
int ret = eepreg_check();
|
|
if (ret == -ENOENT) {
|
|
/* reg does not exist, so make a new one */
|
|
eepreg_reset();
|
|
}
|
|
else if (ret < 0) {
|
|
DEBUG("[eepreg_write] eepreg_check failed\n");
|
|
return ret;
|
|
}
|
|
|
|
/* location of the new data */
|
|
*pos = _get_last_loc(reg_end) - len;
|
|
|
|
if (_new_entry(name, len) < 0) {
|
|
DEBUG("[eepreg_write] not enough space for %s\n", name);
|
|
return -ENOSPC;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int eepreg_rm(const char *name)
|
|
{
|
|
int ret = eepreg_check();
|
|
if (ret < 0) {
|
|
DEBUG("[eepreg_rm] eepreg_check failed\n");
|
|
return ret;
|
|
}
|
|
|
|
uint32_t meta_loc = _get_meta_loc(name);
|
|
|
|
if (meta_loc == (uint32_t)UINT_MAX) {
|
|
DEBUG("[eepreg_rm] no entry for %s\n", name);
|
|
return -ENOENT;
|
|
}
|
|
|
|
uint32_t reg_end = _get_reg_end();
|
|
uint32_t last_loc = _get_last_loc(reg_end);
|
|
|
|
uint8_t meta_len = _get_meta_len(meta_loc);
|
|
uint32_t tot_meta_len = reg_end - meta_loc;
|
|
|
|
uint32_t data_loc = _get_data_loc(meta_loc, meta_len);
|
|
uint32_t data_len = _get_data_len(meta_loc, data_loc);
|
|
|
|
/* data_loc is above last_loc due to descending order */
|
|
uint32_t tot_data_len = data_loc - last_loc;
|
|
|
|
_move_data(meta_loc + meta_len, meta_loc, tot_meta_len);
|
|
|
|
_move_data(last_loc, last_loc + data_len, tot_data_len);
|
|
|
|
reg_end -= meta_len;
|
|
_set_reg_end(reg_end);
|
|
|
|
/* update data locations */
|
|
while (meta_loc < reg_end) {
|
|
meta_len = _get_meta_len(meta_loc);
|
|
data_loc = _get_data_loc(meta_loc, meta_len);
|
|
|
|
/* addition due to descending order */
|
|
_set_data_loc(meta_loc, meta_len, data_loc + data_len);
|
|
|
|
meta_loc += meta_len;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int eepreg_iter(eepreg_iter_cb_t cb, void *arg)
|
|
{
|
|
uint32_t reg_end = _get_reg_end();
|
|
|
|
int ret = eepreg_check();
|
|
if (ret < 0) {
|
|
DEBUG("[eepreg_len] eepreg_check failed\n");
|
|
return ret;
|
|
}
|
|
|
|
uint32_t meta_loc = REG_ENT1_LOC;
|
|
while (meta_loc < reg_end) {
|
|
uint8_t meta_len = _get_meta_len(meta_loc);
|
|
|
|
/* size of memory allocation */
|
|
uint8_t name_len = _calc_name_len(meta_len);
|
|
|
|
char name[name_len + 1];
|
|
|
|
/* terminate string */
|
|
name[name_len] = '\0';
|
|
|
|
_get_name(meta_loc, name, meta_len);
|
|
|
|
/* execute callback */
|
|
ret = cb(name, arg);
|
|
|
|
if (ret < 0) {
|
|
DEBUG("[eepreg_iter] callback reports failure\n");
|
|
return ret;
|
|
}
|
|
|
|
/* only advance if cb didn't delete entry */
|
|
if (_cmp_name(meta_loc, name, meta_len)) {
|
|
meta_loc += meta_len;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int eepreg_check(void)
|
|
{
|
|
char magic[MAGIC_SIZE];
|
|
|
|
/* get magic number from EEPROM */
|
|
if (eeprom_read(REG_MAGIC_LOC, (uint8_t *)magic, MAGIC_SIZE)
|
|
!= MAGIC_SIZE) {
|
|
|
|
DEBUG("[eepreg_check] EEPROM read error\n");
|
|
return -EIO;
|
|
}
|
|
|
|
/* check to see if magic number is the same */
|
|
if (strncmp(magic, eepreg_magic, MAGIC_SIZE) != 0) {
|
|
DEBUG("[eepreg_check] No registry detected\n");
|
|
return -ENOENT;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int eepreg_reset(void)
|
|
{
|
|
/* write new registry magic number */
|
|
if (eeprom_write(REG_MAGIC_LOC, (uint8_t *)eepreg_magic, MAGIC_SIZE)
|
|
!= MAGIC_SIZE) {
|
|
|
|
DEBUG("[eepreg_reset] EEPROM write error\n");
|
|
return -EIO;
|
|
}
|
|
|
|
/* new registry has no entries */
|
|
_set_reg_end(REG_ENT1_LOC);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int eepreg_len(uint32_t *len, const char *name)
|
|
{
|
|
int ret = eepreg_check();
|
|
if (ret < 0) {
|
|
DEBUG("[eepreg_len] eepreg_check failed\n");
|
|
return ret;
|
|
}
|
|
|
|
uint32_t meta_loc = _get_meta_loc(name);
|
|
|
|
if (meta_loc == (uint32_t)UINT_MAX) {
|
|
DEBUG("[eepreg_len] no entry for %s\n", name);
|
|
return -ENOENT;
|
|
}
|
|
|
|
uint32_t data_loc = _get_data_loc(meta_loc, _get_meta_len(meta_loc));
|
|
|
|
*len = _get_data_len(meta_loc, data_loc);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int eepreg_free(uint32_t *len)
|
|
{
|
|
int ret = eepreg_check();
|
|
if (ret < 0) {
|
|
DEBUG("[eepreg_free] eepreg_check failed\n");
|
|
return ret;
|
|
}
|
|
|
|
uint32_t reg_end = _get_reg_end();
|
|
uint32_t last_loc = _get_last_loc(reg_end);
|
|
*len = _calc_free_space(reg_end, last_loc);
|
|
|
|
return 0;
|
|
}
|