diff --git a/sys/cb_mux/Makefile b/sys/cb_mux/Makefile
new file mode 100644
index 0000000000000000000000000000000000000000..48422e909a47d7cd428d10fa73825060ccc8d8c2
--- /dev/null
+++ b/sys/cb_mux/Makefile
@@ -0,0 +1 @@
+include $(RIOTBASE)/Makefile.base
diff --git a/sys/cb_mux/cb_mux.c b/sys/cb_mux/cb_mux.c
new file mode 100644
index 0000000000000000000000000000000000000000..0a8164ebcf7f9de7b45164f1fe429d5e8eac053b
--- /dev/null
+++ b/sys/cb_mux/cb_mux.c
@@ -0,0 +1,124 @@
+/*
+ * 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_cb_mux
+ * @{
+ *
+ * @file
+ * @brief   cb_mux implementation
+ *
+ * @author  Matthew Blue <matthew.blue.neuro@gmail.com>
+ * @}
+ */
+
+#include "cb_mux.h"
+#include "utlist.h"
+
+void cb_mux_add(cb_mux_t **head, cb_mux_t *entry)
+{
+    LL_APPEND(*head, entry);
+}
+
+void cb_mux_del(cb_mux_t **head, cb_mux_t *entry)
+{
+    LL_DELETE(*head, entry);
+}
+
+cb_mux_t *cb_mux_find_cbid(cb_mux_t *head, cb_mux_cbid_t cbid_val)
+{
+    cb_mux_t *entry;
+
+    LL_SEARCH_SCALAR(head, entry, cbid, cbid_val);
+
+    return entry;
+}
+
+cb_mux_t *cb_mux_find_low(cb_mux_t *head)
+{
+    cb_mux_t *entry;
+    cb_mux_t *entry_low = NULL;
+    cb_mux_cbid_t id = (cb_mux_cbid_t)(-1);
+
+    LL_FOREACH(head, entry) {
+        /* Entry does not have lower ID */
+        if (entry->cbid >= id) {
+            continue;
+        }
+
+        id = entry->cbid;
+        entry_low = entry;
+    }
+
+    return entry_low;
+}
+
+cb_mux_t *cb_mux_find_high(cb_mux_t *head)
+{
+    cb_mux_t *entry;
+    cb_mux_t *entry_high = NULL;
+    cb_mux_cbid_t id = 0;
+
+    LL_FOREACH(head, entry) {
+        /* Entry does not have higher ID */
+        if (entry->cbid <= id) {
+            continue;
+        }
+
+        id = entry->cbid;
+        entry_high = entry;
+    }
+
+    return entry_high;
+}
+
+cb_mux_cbid_t cb_mux_find_free_id(cb_mux_t *head)
+{
+    uint32_t free;
+    cb_mux_cbid_t block;
+    cb_mux_t *entry;
+    uint8_t num;
+
+    /* Search for free IDs in blocks of 32 IDs */
+    for (block = 0; block + 31 < (cb_mux_cbid_t)(-1); block += 32) {
+        /* Set all IDs in block to free */
+        free = 0;
+
+        LL_FOREACH(head, entry) {
+            /* cbid falls within this block */
+            if ((entry->cbid >= block) && (entry->cbid < block + 32)) {
+                /* Set cbid to taken */
+                free |= 1 << (entry->cbid & 0x1F);
+            }
+        }
+
+        /* At least one ID in block free */
+        if (~free) {
+            break;
+        }
+    }
+
+    /* Find which ID in block was free */
+    for (num = 0; num < 32; num++) {
+        if (~free & (1 << num)) {
+            return block | num;
+        }
+    }
+
+    /* No free IDs */
+    return (cb_mux_cbid_t)(-1);
+}
+
+void cb_mux_iter(cb_mux_t *head, cb_mux_iter_t func, void *arg)
+{
+    cb_mux_t *entry;
+
+    LL_FOREACH(head, entry) {
+        func(entry, arg);
+    }
+}
diff --git a/sys/include/cb_mux.h b/sys/include/cb_mux.h
new file mode 100644
index 0000000000000000000000000000000000000000..8f28223c105a287e70bcb44a31c17772681e4e0e
--- /dev/null
+++ b/sys/include/cb_mux.h
@@ -0,0 +1,143 @@
+/*
+ * 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_cb_mux Callback multiplexer
+ * @ingroup     sys
+ * @brief       cb_mux provides utilities for storing, retrieving, and managing
+ *              callback information in a singly linked list.
+ *
+ * If an API provides the ability to call multiple callbacks, cb_mux can
+ * simplify handling of an arbitrary number of callbacks by requiring memory
+ * for a cb_mux entry to be passed along with other arguments. The cb_mux entry
+ * is then attached to a list using cb_mux_add. The code implementing that API
+ * can manage the list using the various utility functions that cb_mux provides.
+ *
+ * @{
+ *
+ * @file
+ * @brief       cb_mux interface definitions
+ *
+ * @author      Matthew Blue <matthew.blue.neuro@gmail.com>
+ */
+
+#ifndef CB_MUX_H
+#define CB_MUX_H
+
+#include <stdint.h>
+
+/* For alternate cb_mux_cbid_t */
+#include "periph_cpu.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#ifndef HAVE_CB_MUX_CBID_T
+/**
+ * @brief   cb_mux identifier type
+ */
+typedef unsigned int cb_mux_cbid_t;
+#endif
+
+/**
+ * @brief   cb_mux callback type
+ */
+typedef void (*cb_mux_cb_t)(void *);
+
+/**
+ * @brief   cb_mux list entry structure
+ */
+typedef struct cb_mux {
+    struct cb_mux *next;  /**< next entry in the cb_mux list */
+    cb_mux_cbid_t cbid;   /**< identifier for this callback */
+    void *info;           /**< optional extra information */
+    cb_mux_cb_t cb;       /**< callback function */
+    void *arg;            /**< argument for callback function */
+} cb_mux_t;
+
+/**
+ * @brief   cb_mux iterate function callback type for cb_mux_iter
+ */
+typedef void (*cb_mux_iter_t)(cb_mux_t *, void *);
+
+/**
+ * @brief   Add a new entry to the end of a cb_mux list
+ *
+ * @param[in] head   double pointer to first list entry
+ * @param[in] entry  entry to add
+ */
+void cb_mux_add(cb_mux_t **head, cb_mux_t *entry);
+
+/**
+ * @brief   Remove a entry from a cb_mux list
+ *
+ * @param[in] head   double pointer to first list entry
+ * @param[in] entry  entry to remove
+ */
+void cb_mux_del(cb_mux_t **head, cb_mux_t *entry);
+
+/**
+ * @brief   Find an entry in the list by ID
+ *
+ * @param[in] head      pointer to first list entry
+ * @param[in] cbid_val  ID to find
+ *
+ * @return pointer to the list entry
+ */
+cb_mux_t *cb_mux_find_cbid(cb_mux_t *head, cb_mux_cbid_t cbid_val);
+
+/**
+ * @brief   Find the entry with the lowest ID
+ *
+ * If there are multiple hits, this returns the oldest.
+ *
+ * @param[in] head   pointer to first list entry
+ *
+ * @return pointer to the list entry
+ */
+cb_mux_t *cb_mux_find_low(cb_mux_t *head);
+
+/**
+ * @brief   Find the entry with the highest ID
+ *
+ * If there are multiple hits, this returns the oldest.
+ *
+ * @param[in] head   pointer to first list entry
+ *
+ * @return pointer to the list entry
+ */
+cb_mux_t *cb_mux_find_high(cb_mux_t *head);
+
+/**
+ * @brief   Find the lowest unused ID
+ *
+ * Returns highest possible ID on failure
+ *
+ * @param[in] head   pointer to first list entry
+ *
+ * @return lowest unused ID
+ */
+cb_mux_cbid_t cb_mux_find_free_id(cb_mux_t *head);
+
+/**
+ * @brief   Run a function on every item in the cb_mux list
+ *
+ * @param[in] head  pointer to first list entry
+ * @param[in] func  function to run on each entry
+ * @param[in] arg   argument for the function
+ */
+void cb_mux_iter(cb_mux_t *head, cb_mux_iter_t func, void *arg);
+
+#ifdef __cplusplus
+}
+#endif
+
+/** @} */
+
+#endif /* CB_MUX_H */
diff --git a/tests/cb_mux/Makefile b/tests/cb_mux/Makefile
new file mode 100644
index 0000000000000000000000000000000000000000..85812a0d9df5bab77d7e723b10eca1bd02ece86a
--- /dev/null
+++ b/tests/cb_mux/Makefile
@@ -0,0 +1,10 @@
+include ../Makefile.tests_common
+
+USEMODULE += cb_mux
+
+TEST_ON_CI_WHITELIST += all
+
+include $(RIOTBASE)/Makefile.include
+
+test:
+	./tests/01-run.py
diff --git a/tests/cb_mux/main.c b/tests/cb_mux/main.c
new file mode 100644
index 0000000000000000000000000000000000000000..872834ed7b5710fd143f963d76bfff6ea1fcd992
--- /dev/null
+++ b/tests/cb_mux/main.c
@@ -0,0 +1,135 @@
+/*
+ * 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       cb_mux test application
+ *
+ * @author      Matthew Blue <matthew.blue.neuro@gmail.com>
+ * @}
+ */
+
+#include <stdio.h>
+
+#include "cb_mux.h"
+
+/* Head of cb_mux list */
+cb_mux_t *cb_mux_head;
+
+/* Flags for mux_iter */
+enum {
+    ITER_TEST = 1
+};
+
+/* Function to iterate over cb_mux list */
+void mux_iter(cb_mux_t *entry, void *arg)
+{
+    (void)arg;
+
+    entry->info = (void *)((uintptr_t)entry->info | (1 << ITER_TEST));
+}
+
+/* Test callback */
+void cb(void *arg)
+{
+    printf("Callback %u executed\n", (uint8_t)(uintptr_t)arg);
+}
+
+int main(void)
+{
+    cb_mux_t entries[5];
+    cb_mux_cbid_t num;
+    cb_mux_t *entry;
+
+    puts("cb_mux test routine");
+
+    for (num = 0; num < 5; num++) {
+        entries[num].cb = cb;
+        entries[num].arg = (void *)num;
+        entries[num].cbid = num;
+    }
+
+    puts("Test list addition, retrieval, execution of 5 CBs");
+
+    for (num = 0; num < 5; num++) {
+        cb_mux_add(&cb_mux_head, &(entries[num]));
+    }
+
+    for (num = 0; num < 5; num++) {
+        entry = cb_mux_find_cbid(cb_mux_head, num);
+
+        if (entry->cb == NULL) {
+            puts("[FAILED] Unexpected NULL pointer");
+            return 1;
+        }
+
+        entry->cb(entry->arg);
+    }
+
+    puts("Test list deletion of CB 0, 2, 4, execution of 1, 3");
+
+    cb_mux_del(&cb_mux_head, &(entries[0]));
+    cb_mux_del(&cb_mux_head, &(entries[2]));
+    cb_mux_del(&cb_mux_head, &(entries[4]));
+
+    for (num = 0; num < 5; num++) {
+        entry = cb_mux_find_cbid(cb_mux_head, num);
+
+        if (entry == NULL) {
+            continue;
+        }
+
+        entry->cb(entry->arg);
+    }
+
+    puts("Test execution of CB with lowest ID (1)");
+
+    entry = cb_mux_find_low(cb_mux_head);
+    if (entry->cb == NULL) {
+        puts("[FAILED] Unexpected NULL pointer");
+        return 1;
+    }
+    entry->cb(entry->arg);
+
+    puts("Test execution of CB with highest ID (3)");
+
+    entry = cb_mux_find_high(cb_mux_head);
+    if (entry->cb == NULL) {
+        puts("[FAILED] Unexpected NULL pointer");
+        return 1;
+    }
+    entry->cb(entry->arg);
+
+    puts("Re-adding list entries (0, 2, 4) by finding next free ID");
+
+    num = cb_mux_find_free_id(cb_mux_head);
+    while (num < 5) {
+        cb_mux_add(&cb_mux_head, &(entries[num]));
+
+        printf("Added entry %i\n", num);
+
+        num = cb_mux_find_free_id(cb_mux_head);
+    }
+
+    puts("Test iteration of a function over list");
+
+    cb_mux_iter(cb_mux_head, mux_iter, NULL);
+
+    for (num = 0; num < 5; num++) {
+        if ((uintptr_t)entries[num].info & (1 << ITER_TEST)) {
+            printf("Entry %i was updated correctly\n", num);
+        }
+    }
+
+    puts("Tests complete!");
+
+    return 0;
+}
diff --git a/tests/cb_mux/tests/01-run.py b/tests/cb_mux/tests/01-run.py
new file mode 100755
index 0000000000000000000000000000000000000000..d501e26601605e988f398fbe9d6b3f7bacb5e5e6
--- /dev/null
+++ b/tests/cb_mux/tests/01-run.py
@@ -0,0 +1,44 @@
+#!/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 os
+import sys
+
+
+def testfunc(child):
+    child.expect_exact("cb_mux test routine")
+    child.expect_exact("Test list addition, retrieval, execution of 5 CBs")
+    child.expect_exact("Callback 0 executed")
+    child.expect_exact("Callback 1 executed")
+    child.expect_exact("Callback 2 executed")
+    child.expect_exact("Callback 3 executed")
+    child.expect_exact("Callback 4 executed")
+    child.expect_exact("Test list deletion of CB 0, 2, 4, execution of 1, 3")
+    child.expect_exact("Callback 1 executed")
+    child.expect_exact("Callback 3 executed")
+    child.expect_exact("Test execution of CB with lowest ID (1)")
+    child.expect_exact("Callback 1 executed")
+    child.expect_exact("Test execution of CB with highest ID (3)")
+    child.expect_exact("Callback 3 executed")
+    child.expect_exact("Re-adding list entries (0, 2, 4) by finding next free ID")
+    child.expect_exact("Added entry 0")
+    child.expect_exact("Added entry 2")
+    child.expect_exact("Added entry 4")
+    child.expect_exact("Test iteration of a function over list")
+    child.expect_exact("Entry 0 was updated correctly")
+    child.expect_exact("Entry 1 was updated correctly")
+    child.expect_exact("Entry 2 was updated correctly")
+    child.expect_exact("Entry 3 was updated correctly")
+    child.expect_exact("Entry 4 was updated correctly")
+    child.expect_exact("Tests complete!")
+
+
+if __name__ == "__main__":
+    sys.path.append(os.path.join(os.environ['RIOTBASE'], 'dist/tools/testrunner'))
+    from testrunner import run
+    sys.exit(run(testfunc))
diff --git a/tests/cb_mux_bench/Makefile b/tests/cb_mux_bench/Makefile
new file mode 100644
index 0000000000000000000000000000000000000000..61b37720f2a84d916860a766ddc1c35b79d1f96c
--- /dev/null
+++ b/tests/cb_mux_bench/Makefile
@@ -0,0 +1,11 @@
+include ../Makefile.tests_common
+
+USEMODULE += cb_mux \
+             xtimer
+
+TEST_ON_CI_WHITELIST += all
+
+include $(RIOTBASE)/Makefile.include
+
+test:
+	./tests/01-run.py
diff --git a/tests/cb_mux_bench/main.c b/tests/cb_mux_bench/main.c
new file mode 100644
index 0000000000000000000000000000000000000000..478d4f1657d20d2c863a941adee86a1721ac1f39
--- /dev/null
+++ b/tests/cb_mux_bench/main.c
@@ -0,0 +1,90 @@
+/*
+ * 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       cb_mux benchmark application
+ *
+ * @author      Matthew Blue <matthew.blue.neuro@gmail.com>
+ * @}
+ */
+
+#include <stdio.h>
+
+#include "cb_mux.h"
+#include "xtimer.h"
+
+/* Number of entries in the cb_mux list */
+#define NUM_ENTRIES (20U)
+
+/* Fail if us greater than threshold */
+#define FAIL_THRESH (200UL)
+
+/* Head of cb_mux list */
+cb_mux_t *cb_mux_head;
+
+/* cb_mux list entries */
+cb_mux_t entries[NUM_ENTRIES];
+
+/* Timing */
+unsigned long time_prev, time_curr;
+
+void cb(void *arg)
+{
+    (void)arg;
+    time_curr = xtimer_now_usec();
+}
+
+int main(void)
+{
+    unsigned long xtimer_delay, time_diff;
+    uint8_t num;
+    cb_mux_t *entry;
+
+    puts("cb_mux benchmark application");
+
+    /* Delay due to fetching timer with xtimer */
+    time_prev = xtimer_now_usec();
+    xtimer_delay = time_prev - xtimer_now_usec();
+
+    /* Test for worst case: finding last entry */
+    entries[NUM_ENTRIES - 1].cbid = 1;
+
+    printf("Populating cb_mux list with %u items\n", NUM_ENTRIES);
+
+    for (num = 0; num < NUM_ENTRIES; num++) {
+        entries[num].cb = cb;
+        cb_mux_add(&cb_mux_head, &(entries[num]));
+    }
+
+    puts("Finding the last list entry");
+
+    time_prev = xtimer_now_usec();
+
+    entry = cb_mux_find_cbid(cb_mux_head, 1);
+    entry->cb(entry->arg);
+
+    time_diff = time_curr - time_prev - xtimer_delay;
+
+    printf("List walk time: %lu us\n", time_diff);
+
+    if (time_diff > FAIL_THRESH) {
+        printf("Walk time greater than threshold of %lu us\n", FAIL_THRESH);
+        puts("[FAILURE]");
+        return 1;
+    }
+    else {
+        printf("Walk time less than threshold of %lu us\n", FAIL_THRESH);
+        puts("[SUCCESS]");
+    }
+
+    return 0;
+}
diff --git a/tests/cb_mux_bench/tests/01-run.py b/tests/cb_mux_bench/tests/01-run.py
new file mode 100755
index 0000000000000000000000000000000000000000..23896a9dfe4a0e1678d3bc492324d7a527a10f5a
--- /dev/null
+++ b/tests/cb_mux_bench/tests/01-run.py
@@ -0,0 +1,25 @@
+#!/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 os
+import sys
+
+
+def testfunc(child):
+    child.expect_exact("cb_mux benchmark application")
+    child.expect(u"Populating cb_mux list with \d+ items")
+    child.expect_exact("Finding the last list entry")
+    child.expect(u"List walk time: \d+ us")
+    child.expect(u"Walk time less than threshold of \d+ us")
+    child.expect_exact("[SUCCESS]")
+
+
+if __name__ == "__main__":
+    sys.path.append(os.path.join(os.environ['RIOTBASE'], 'dist/tools/testrunner'))
+    from testrunner import run
+    sys.exit(run(testfunc))