diff --git a/sys/Makefile b/sys/Makefile
index 6068eb76e3223c519e54a3085859734e6f843be2..bab6d947837d8e43f9d67dfab3059edd5fcc882e 100644
--- a/sys/Makefile
+++ b/sys/Makefile
@@ -1,6 +1,9 @@
 ifneq (,$(filter csma_sender,$(USEMODULE)))
   DIRS += net/link_layer/csma_sender
 endif
+ifneq (,$(filter eepreg,$(USEMODULE)))
+  DIRS += eepreg
+endif
 ifneq (,$(filter posix_semaphore,$(USEMODULE)))
   DIRS += posix/semaphore
 endif
diff --git a/sys/Makefile.dep b/sys/Makefile.dep
index b2e06357c4afda528ad3da368e97b3018557cf3c..ce7f6f2c8296f961fe28ef2cf5518751d4773bd3 100644
--- a/sys/Makefile.dep
+++ b/sys/Makefile.dep
@@ -1,3 +1,7 @@
+ifneq (,$(filter eepreg,$(USEMODULE)))
+  FEATURES_REQUIRED += periph_eeprom
+endif
+
 ifneq (,$(filter prng_fortuna,$(USEMODULE)))
   CFLAGS += -DCRYPTO_AES
 endif
diff --git a/sys/eepreg/Makefile b/sys/eepreg/Makefile
new file mode 100644
index 0000000000000000000000000000000000000000..48422e909a47d7cd428d10fa73825060ccc8d8c2
--- /dev/null
+++ b/sys/eepreg/Makefile
@@ -0,0 +1 @@
+include $(RIOTBASE)/Makefile.base
diff --git a/sys/eepreg/eepreg.c b/sys/eepreg/eepreg.c
new file mode 100644
index 0000000000000000000000000000000000000000..43d63059164e488a5574b6d0e9ff9772f349a2b4
--- /dev/null
+++ b/sys/eepreg/eepreg.c
@@ -0,0 +1,498 @@
+/*
+ * 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;
+}
diff --git a/sys/include/eepreg.h b/sys/include/eepreg.h
new file mode 100644
index 0000000000000000000000000000000000000000..269d6c3044bdcc25ea64d603a3f423c748ded224
--- /dev/null
+++ b/sys/include/eepreg.h
@@ -0,0 +1,271 @@
+/*
+ * 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.
+ */
+
+/**
+ * @defgroup    sys_eepreg EEPROM registration
+ * @ingroup     sys
+ * @brief       eepreg provides a facility to easily manage the locations of
+ *              data stored in EEPROM via a meta-data registry.
+ *
+ * The structure of the meta-data registry is intended to make it easy to
+ * detect the exact layout of existent data so that automatic tools may be
+ * written to migrate legacy data to new formats. It also allows the addition
+ * and removal of new entries dynamically.
+ *
+ * @note   Names are used as identifiers and must be unique! It is also
+ * recommended to keep them as short as possible (while still being unique and
+ * human readable), as many systems have very small amounts of EEPROM.
+ * Disemvowelment can shorten long names while still retaining readability.
+ *
+ * @code {unparsed}
+ * The layout of the EEPROM used looks like this:
+ *    EEPROM_RESERV_CPU_LOW
+ *    EEPROM_RESERV_BOARD_LOW
+ *    Registry magic number ("RIOTREG")
+ *    Registry end pointer
+ *    Registry entry 1 meta-data length (1 byte)
+ *    Registry entry 1 name (unterminated)
+ *    Registry entry 1 data pointer
+ *    Registry entry 2 meta-data length
+ *    Registry entry 2 name
+ *    Registry entry 2 data pointer
+ *    ... (new registry meta-data may be added in ascending order)
+ *    unused space
+ *    ... (new data locations may be added in descending order)
+ *    Entry 2 data
+ *    Entry 1 data
+ *    EEPROM_RESERV_BOARD_HI
+ *    EEPROM_RESERV_CPU_HI
+ * @endcode
+ *
+ * Pointer length is dependent on the size of the available EEPROM (see
+ * EEPREG_PTR_LEN below).
+ *
+ * @{
+ *
+ * @file
+ * @brief       eepreg interface definitions
+ *
+ * @author      Matthew Blue <matthew.blue.neuro@gmail.com>
+ */
+
+#ifndef EEPREG_H
+#define EEPREG_H
+
+#include <stdint.h>
+
+#include "periph_cpu.h"
+#include "periph_conf.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#ifndef EEPROM_RESERV_CPU_LOW
+/**
+ * @brief   EEPROM reserved near beginning for use by CPU and related
+ *
+ * Change with care, as it may make existing data difficult to migrate
+ */
+#define EEPROM_RESERV_CPU_LOW    (0U)
+#endif
+
+#ifndef EEPROM_RESERV_CPU_HI
+/**
+ * @brief   EEPROM reserved near end for use by CPU and related
+ *
+ * Change with care, as it may make existing data difficult to migrate
+ */
+#define EEPROM_RESERV_CPU_HI    (0U)
+#endif
+
+#ifndef EEPROM_RESERV_BOARD_LOW
+/**
+ * @brief   EEPROM reserved near beginning for use by board and related
+ *
+ * Change with care, as it may make existing data difficult to migrate
+ */
+#define EEPROM_RESERV_BOARD_LOW    (0U)
+#endif
+
+#ifndef EEPROM_RESERV_BOARD_HI
+/**
+ * @brief   EEPROM reserved near end for use by board and related
+ *
+ * Change with care, as it may make existing data difficult to migrate
+ */
+#define EEPROM_RESERV_BOARD_HI    (0U)
+#endif
+
+/**
+ * @brief   Size in bytes of pointer meta-data in EEPROM
+ */
+#if (EEPROM_SIZE > 0x1000000)
+#define EEPREG_PTR_LEN    (4U)
+#elif (EEPROM_SIZE > 0x10000)
+#define EEPREG_PTR_LEN    (3U)
+#elif (EEPROM_SIZE > 0x100)
+#define EEPREG_PTR_LEN    (2U)
+#else
+#define EEPREG_PTR_LEN    (1U)
+#endif
+
+/**
+ * @brief   Signature of callback for iterating over entries in EEPROM registry
+ *
+ * @param[in] name    name of an entry in the registry
+ * @param[in] arg     argument for cb
+ *
+ * @return    0 on success
+ * @return    < 0 on failure
+ */
+typedef int (*eepreg_iter_cb_t)(char *name, void *arg);
+
+/**
+ * @brief   Load or write meta-data in EEPROM registry
+ *
+ * This checks to see if relevant meta-data exists in the EEPROM registry, and
+ * returns that data position if it exists. If an entry does not exist in the
+ * registry, meta-data is written and allocated data space if there is enough
+ * remaining. Requesting a different length for an existent entry returns an
+ * error.
+ *
+ * @param[out] pos    pointer to position variable
+ * @param[in] name    name of entry to load or write
+ * @param[in] len     requested amount of data storage
+ *
+ * @return    0 on success
+ * @return    -EIO on EEPROM I/O error
+ * @return    -ENOSPC on insufficient EEPROM for entry
+ * @return    -EADDRINUSE on existing entry with different length
+ */
+int eepreg_add(uint32_t *pos, const char *name, uint32_t len);
+
+/**
+ * @brief   Read position meta-data from EEPROM registry
+ *
+ * This is similar to eepreg_add, except it never writes meta-data.
+ *
+ * @param[out] pos    pointer to position variable
+ * @param[in] name    name of entry to load
+ *
+ * @return    0 on success
+ * @return    -EIO on EEPROM I/O error
+ * @return    -ENOENT on non-existent registry or entry
+ */
+int eepreg_read(uint32_t *pos, const char *name);
+
+/**
+ * @brief   Write meta-data to EEPROM registry
+ *
+ * This ignores existing meta-data and always makes a new entry in the
+ * registry. Typical use should be through eepreg_add and not eepreg_write.
+ * If multiple entries with the same name exist, eepreg functions will find
+ * the oldest. Mainly intended for use by migration utilities.
+ *
+ * @param[out] pos    pointer to position variable
+ * @param[in] name    name of entry to write
+ * @param[in] len     requested amount of data storage
+ *
+ * @return    0 on success
+ * @return    -EIO on EEPROM I/O error
+ * @return    -ENOSPC on insufficient EEPROM for entry
+ */
+int eepreg_write(uint32_t *pos, const char *name, uint32_t len);
+
+/**
+ * @brief   Remove entry from EEPROM registry and free space
+ *
+ * This removes an entry from the EEPROM registry and its corresponding data
+ * and moves the data and meta-data of entries after removed entry to occupy
+ * the freed space. This preserves the structure of the EEPROM registry.
+ * Warning: this is a read/write intensive operation! Mainly intended for use
+ * by migration utilities.
+ *
+ * @param[in] name    name of entry to remove
+ *
+ * @return    0 on success
+ * @return    -EIO on EEPROM I/O error
+ * @return    -ENOENT on non-existent registry or entry
+ */
+int eepreg_rm(const char *name);
+
+/**
+ * @brief   Iterate over meta-data entries in EEPROM registry
+ *
+ * This executes a callback over each name in the EEPROM registry. The intended
+ * work-flow for migration is to: iterate over each entry, check to see if
+ * migration is needed, duplicate using eepreg_write if needed, migrate data to
+ * duplicate entry, then delete old entry using eepreg_rm.
+ *
+ * @note   It is safe for the callback to remove the entry it is called with,
+ * or to add new entries.
+ *
+ * @param[in] cb     callback to iterate over entries
+ * @param[in] arg    argument for cb
+ *
+ * @return    0 on success
+ * @return    -EIO on EEPROM I/O error
+ * @return    -ENOENT on non-existent registry
+ * @return    return value of cb when cb returns < 0
+ */
+int eepreg_iter(eepreg_iter_cb_t cb, void *arg);
+
+/**
+ * @brief   Check for the presence of meta-data registry
+ *
+ * @return    0 on success
+ * @return    -EIO on EEPROM I/O error
+ * @return    -ENOENT on non-existent registry
+ */
+int eepreg_check(void);
+
+/**
+ * @brief   Clear existing meta-data registry
+ *
+ * This removes any existing meta-data registry by writing a new registry with
+ * no entries.
+ *
+ * @return    0 on success
+ * @return    -EIO on EEPROM I/O error
+ */
+int eepreg_reset(void);
+
+/**
+ * @brief   Calculate data length from meta-data in EEPROM registry
+ *
+ * @note   This information is typically already available to code that has
+ * called eepreg_add.
+ *
+ * @param[out] len    pointer to length variable
+ * @param[in] name    name of entry to load or write
+ *
+ * @return    0 on success
+ * @return    -EIO on EEPROM I/O error
+ * @return    -ENOENT on non-existent registry or entry
+ */
+int eepreg_len(uint32_t *len, const char *name);
+
+/**
+ * @brief   Calculate length of remaining EEPROM free space
+ *
+ * @param[out] len    pointer to length variable
+ *
+ * @return    0 on success
+ * @return    -EIO on EEPROM I/O error
+ * @return    -ENOENT on non-existent registry
+ */
+int eepreg_free(uint32_t *len);
+
+#ifdef __cplusplus
+}
+#endif
+
+/** @} */
+
+#endif /* EEPREG_H */
diff --git a/tests/eepreg/Makefile b/tests/eepreg/Makefile
new file mode 100644
index 0000000000000000000000000000000000000000..12b93f03741f75bb5205d4627de60b3db7195679
--- /dev/null
+++ b/tests/eepreg/Makefile
@@ -0,0 +1,7 @@
+include ../Makefile.tests_common
+
+FEATURES_REQUIRED = periph_eeprom
+
+USEMODULE += eepreg
+
+include $(RIOTBASE)/Makefile.include
diff --git a/tests/eepreg/README.md b/tests/eepreg/README.md
new file mode 100644
index 0000000000000000000000000000000000000000..e53b28f005c45d25a6abbfa05d4a623df1b005ba
--- /dev/null
+++ b/tests/eepreg/README.md
@@ -0,0 +1,6 @@
+# EEPROM registry (eepreg) test
+
+This test will verify the functionality of the eepreg module.
+
+WARNING: This will write to your EEPROM and probably corrupt any data that is
+there!
diff --git a/tests/eepreg/main.c b/tests/eepreg/main.c
new file mode 100644
index 0000000000000000000000000000000000000000..b56f24df3f76a4cf051b1b38dfa6da5cd96c646b
--- /dev/null
+++ b/tests/eepreg/main.c
@@ -0,0 +1,203 @@
+/*
+ * 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     tests
+ * @{
+ *
+ * @file
+ * @brief       eepreg test application
+ *
+ * @author      Matthew Blue <matthew.blue.neuro@gmail.com>
+ * @}
+ */
+
+#include <errno.h>
+#include <stdint.h>
+#include <stdio.h>
+#include <string.h>
+
+#include "eepreg.h"
+#include "periph/eeprom.h"
+
+#define DAT_START    (EEPROM_SIZE - EEPROM_RESERV_CPU_HI \
+                      - EEPROM_RESERV_BOARD_HI - 1)
+
+#define ENT1_NAME    "foo"
+#define ENT1_SIZE    (12U)
+#define ENT2_NAME    "bar"
+#define ENT2_SIZE    (34U)
+#define DATA         "spam and eggs"
+
+int eepreg_iter_cb(char *name, void *arg)
+{
+    (void)arg;
+
+    printf("%s ", name);
+
+    return 0;
+}
+
+int main(void)
+{
+    int ret;
+    uint32_t tmp1, tmp2, tmp3;
+    char data[sizeof(DATA)];
+
+    puts("EEPROM registry (eepreg) test routine");
+
+    printf("Testing new registry creation: ");
+
+    printf("reset ");
+    ret = eepreg_reset();
+    if (ret < 0) {
+        puts("[FAILED]");
+        return 1;
+    }
+
+    printf("check ");
+    ret = eepreg_check();
+    if (ret < 0) {
+        puts("[FAILED]");
+        return 1;
+    }
+
+    puts("[SUCCESS]");
+
+    printf("Testing writing and reading entries: ");
+
+    printf("add ");
+    ret = eepreg_add(&tmp1, ENT1_NAME, ENT1_SIZE);
+    if (ret < 0 || tmp1 != DAT_START - ENT1_SIZE) {
+        puts("[FAILED]");
+        return 1;
+    }
+
+    printf("write ");
+    ret = eepreg_write(&tmp2, ENT2_NAME, ENT2_SIZE);
+    if (ret < 0 || tmp2 != DAT_START - ENT1_SIZE - ENT2_SIZE) {
+        puts("[FAILED]");
+        return 1;
+    }
+
+    /* read via add */
+    printf("add ");
+    ret = eepreg_add(&tmp3, ENT1_NAME, ENT1_SIZE);
+    if (ret < 0 || tmp1 != tmp3) {
+        puts("[FAILED]");
+        return 1;
+    }
+
+    printf("read ");
+    ret = eepreg_read(&tmp1, ENT2_NAME);
+    if (ret < 0 || tmp1 != tmp2) {
+        puts("[FAILED]");
+        return 1;
+    }
+
+    puts("[SUCCESS]");
+
+    printf("Testing detection of conflicting size: ");
+
+    printf("add ");
+    ret = eepreg_add(&tmp1, ENT1_NAME, ENT1_SIZE + 1);
+    if (ret != -EADDRINUSE) {
+        puts("[FAILED]");
+        return 1;
+    }
+
+    puts("[SUCCESS]");
+
+    printf("Testing calculation of lengths: ");
+
+    printf("len ");
+    ret = eepreg_len(&tmp1, ENT1_NAME);
+    if (ret < 0 || tmp1 != ENT1_SIZE) {
+        puts("[FAILED]");
+        return 1;
+    }
+
+    printf("len ");
+    ret = eepreg_len(&tmp2, ENT2_NAME);
+    if (ret < 0 || tmp2 != ENT2_SIZE) {
+        puts("[FAILED]");
+        return 1;
+    }
+
+    puts("[SUCCESS]");
+
+    printf("Testing of successful data move after rm: ");
+
+    printf("rm ");
+    eepreg_read(&tmp1, ENT1_NAME);
+    eepreg_read(&tmp2, ENT2_NAME);
+    eeprom_write(tmp2, (uint8_t *)DATA, sizeof(DATA));
+    ret = eepreg_rm(ENT1_NAME);
+    if (ret < 0) {
+        puts("[FAILED]");
+        return 1;
+    }
+
+    printf("read ");
+    ret = eepreg_read(&tmp3, ENT2_NAME);
+    if (ret < 0 || tmp3 != (DAT_START - ENT2_SIZE)) {
+        puts("[FAILED]");
+        return 1;
+    }
+
+    printf("data ");
+    eeprom_read(tmp3, (uint8_t *)data, sizeof(DATA));
+    if (strcmp(data, DATA) != 0) {
+        puts("[FAILED]");
+        return 1;
+    }
+
+    puts("[SUCCESS]");
+
+    printf("Testing of free space change after write: ");
+
+    printf("free ");
+    ret = eepreg_free(&tmp1);
+    if (ret < 0) {
+        puts("[FAILED]");
+        return 1;
+    }
+
+    printf("add ");
+    ret = eepreg_add(&tmp3, ENT1_NAME, ENT1_SIZE);
+    if (ret < 0) {
+        puts("[FAILED]");
+        return 1;
+    }
+
+    printf("free ");
+    ret = eepreg_free(&tmp2);
+    if (ret < 0
+        || tmp1 != (tmp2 + ENT1_SIZE + sizeof(ENT1_NAME) + EEPREG_PTR_LEN)) {
+
+        puts("[FAILED]");
+        return 1;
+    }
+
+    puts("[SUCCESS]");
+
+    printf("Testing of iteration over registry: ");
+
+    printf("iter ");
+    ret = eepreg_iter(eepreg_iter_cb, NULL);
+    if (ret < 0) {
+        puts("[FAILED]");
+        return 1;
+    }
+
+    puts("[SUCCESS]");
+
+    puts("Tests complete!");
+
+    return 0;
+}
diff --git a/tests/eepreg/tests/01-run.py b/tests/eepreg/tests/01-run.py
new file mode 100755
index 0000000000000000000000000000000000000000..d26ab489ab86f5c6772beffa6028f87adeb4a46a
--- /dev/null
+++ b/tests/eepreg/tests/01-run.py
@@ -0,0 +1,26 @@
+#!/usr/bin/env python3
+
+# 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.
+
+import sys
+from testrunner import run
+
+
+def testfunc(child):
+    child.expect_exact("EEPROM registry (eepreg) test routine")
+    child.expect_exact("Testing new registry creation: reset check [SUCCESS]")
+    child.expect_exact("Testing writing and reading entries: add write add read [SUCCESS]")
+    child.expect_exact("Testing detection of conflicting size: add [SUCCESS]")
+    child.expect_exact("Testing calculation of lengths: len len [SUCCESS]")
+    child.expect_exact("Testing of successful data move after rm: rm read data [SUCCESS]")
+    child.expect_exact("Testing of free space change after write: free add free [SUCCESS]")
+    child.expect_exact("Testing of iteration over registry: iter bar foo [SUCCESS]")
+    child.expect_exact("Tests complete!")
+
+
+if __name__ == "__main__":
+    sys.exit(run(testfunc))