diff --git a/sys/ecc/golay2412.c b/sys/ecc/golay2412.c
new file mode 100644
index 0000000000000000000000000000000000000000..6f18346b600eabb22c6b0e2d2886174ab4688f02
--- /dev/null
+++ b/sys/ecc/golay2412.c
@@ -0,0 +1,427 @@
+/*
+ * Copyright (c) 2007 - 2015 Joseph Gaeddert
+ * Copyright (c) 2018        HAW Hamburg
+ *
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
+ * of this software and associated documentation files (the "Software"), to deal
+ * in the Software without restriction, including without limitation the rights
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+ * copies of the Software, and to permit persons to whom the Software is
+ * furnished to do so, subject to the following conditions:
+ *
+ * The above copyright notice and this permission notice shall be included in
+ * all copies or substantial portions of the Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+ * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+ * THE SOFTWARE.
+ */
+
+/**
+ * @ingroup     sys_ecc
+ * @{
+ *
+ * @file
+ * @brief       Golay(24,12) half-rate forward error-correction code
+ *
+ * Code referring to Liquid-DSP
+ * https://github.com/jgaeddert/liquid-dsp/blob/master/src/fec/src/fec_golay2412.c
+ *
+ * @author      Joseph Gaeddert
+ * @author      Peter Kietzmann <peter.kietzmann@haw.hamburg.de>
+ *
+ * @}
+ */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <assert.h>
+#include <errno.h>
+
+#include "bitarithm.h"
+#include "ecc/golay2412.h"
+
+#define ENABLE_DEBUG        (0)
+#include "debug.h"
+
+#if ENABLE_DEBUG
+#define DEBUG_FEC_GOLAY2412 (1)
+#else
+#define DEBUG_FEC_GOLAY2412 (0)
+#endif
+
+/* generator matrix transposed [24 x 12] */
+static const uint32_t golay2412_Gt[24] = {
+    0x08ed, 0x01db, 0x03b5, 0x0769, 0x0ed1, 0x0da3, 0x0b47, 0x068f,
+    0x0d1d, 0x0a3b, 0x0477, 0x0ffe, 0x0800, 0x0400, 0x0200, 0x0100,
+    0x0080, 0x0040, 0x0020, 0x0010, 0x0008, 0x0004, 0x0002, 0x0001
+};
+
+/* P matrix [12 x 12] */
+static const uint32_t golay2412_P[12] = {
+    0x08ed, 0x01db, 0x03b5, 0x0769,
+    0x0ed1, 0x0da3, 0x0b47, 0x068f,
+    0x0d1d, 0x0a3b, 0x0477, 0x0ffe
+};
+
+/* parity check matrix [12 x 24] */
+static const uint32_t golay2412_H[12] = {
+    0x008008ed, 0x004001db, 0x002003b5, 0x00100769,
+    0x00080ed1, 0x00040da3, 0x00020b47, 0x0001068f,
+    0x00008d1d, 0x00004a3b, 0x00002477, 0x00001ffe
+};
+
+#if DEBUG_FEC_GOLAY2412
+/* print string of bits to standard output */
+static inline void liquid_print_bitstring(uint32_t _x,
+                                          uint32_t _n)
+{
+    uint32_t i;
+
+    for (i = 0; i < _n; i++) {
+        printf("%" PRIu32, (_x >> (_n - i - 1)) & 1);
+    }
+}
+#endif
+
+#ifndef NDEBUG
+static uint32_t block_get_enc_msg_len(uint32_t _dec_msg_len,
+                                      uint32_t _m,
+                                      uint32_t _k)
+{
+    assert((_m > 0) && (_k >= _m));
+
+    /* compute total number of bits in decoded message  */
+    uint32_t num_bits_in = _dec_msg_len * 8;
+
+    /* compute total number of blocks: ceil(num_bits_in/_m) */
+    uint32_t num_blocks = num_bits_in / _m + ((num_bits_in % _m) ? 1 : 0);
+
+    /* compute total number of bits out */
+    uint32_t num_bits_out = num_blocks * _k;
+
+    /* compute total number of bytes out: ceil(num_bits_out/8) */
+    uint32_t num_bytes_out = num_bits_out / 8 + ((num_bits_out % 8) ? 1 : 0);
+
+#if DEBUG_FEC_GOLAY2412
+    printf("block_get_enc_msg_len(%" PRIu32 ",%" PRIu32 ",%" PRIu32 ")\n", _dec_msg_len, _m, _k);
+    printf("    dec msg len :   %" PRIu32 " bytes\n", _dec_msg_len);
+    printf("    m           :   %" PRIu32 " bits\n", _m);
+    printf("    k           :   %" PRIu32 " bits\n", _k);
+    printf("    num bits in :   %" PRIu32 " bits\n", num_bits_in);
+    printf("    num blocks  :   %" PRIu32 "\n", num_blocks);
+    printf("    num bits out:   %" PRIu32 " bits\n", num_bits_out);
+    printf("    enc msg len :   %" PRIu32 " bytes\n", num_bytes_out);
+#endif
+
+    return num_bytes_out;
+}
+#endif
+
+/* multiply input vector with parity check matrix, H */
+static uint32_t golay2412_matrix_mul(uint32_t _v,
+                                     const uint32_t *_A,
+                                     uint32_t _n)
+{
+    uint32_t x = 0;
+    uint32_t i;
+
+    for (i = 0; i < _n; i++) {
+        x <<= 1;
+        /* same as above, but exploit the fact that vectors are at
+         * most 24 bits long */
+        uint32_t p = _A[i] & _v;
+        uint32_t c = 0;
+        c = bitarithm_bits_set_u32(p & 0x00ffffff);
+
+        /* mod 2 */
+        x |= c & 0x0001;
+    }
+    return x;
+}
+static uint32_t golay2412_encode_symbol(uint32_t _sym_dec, const uint32_t *_A)
+{
+    /* validate input */
+    assert(_sym_dec > (1 << 12));
+
+    /* compute encoded/transmitted message: v = m*G */
+    return golay2412_matrix_mul(_sym_dec, _A, 24);
+}
+
+/* search for p[i] such that w(v+p[i]) <= 2, return -1 on fail */
+static int8_t golay2412_parity_search(uint32_t _v, const uint32_t *_A)
+{
+    assert(_v < (1 << 12));
+
+    uint8_t i;
+    for (i = 0; i < 12; i++) {
+        /* same as above but faster, exploiting fact that P has
+         * only 12 bits of resolution */
+        uint32_t p = _v ^ _A[i];
+
+        if (bitarithm_bits_set_u32(p & 0x00000fff) <= 2) {
+            return i;
+        }
+    }
+
+    /* could not find p[i] to satisfy criteria */
+    return -1;
+}
+
+static uint32_t golay2412_decode_symbol(uint32_t _sym_enc,
+                                        const uint32_t *_A,
+                                        const uint32_t *_B)
+{
+    /* validate input */
+    assert((_sym_enc) < (1L << 24));
+
+    /* state variables */
+    uint32_t s = 0;         /* syndrome vector */
+    uint32_t e_hat = 0;     /* estimated error vector */
+    uint32_t v_hat = 0;     /* estimated transmitted message */
+    uint32_t m_hat = 0;     /* estimated original message */
+
+    /* compute syndrome vector, s = r*H^T = ( H*r^T )^T */
+    s = golay2412_matrix_mul(_sym_enc, _B, 12);
+#if DEBUG_FEC_GOLAY2412
+    printf("s (syndrome vector): "); liquid_print_bitstring(s, 12); printf("\n");
+#endif
+
+    /* compute weight of s (12 bits) */
+    uint8_t ws = bitarithm_bits_set_u32(s & 0x00000fff);
+#if DEBUG_FEC_GOLAY2412
+    printf("w(s) = %u\n", ws);
+#endif
+
+    /* step 2: */
+    e_hat = 0;
+    if (ws <= 3) {
+#if DEBUG_FEC_GOLAY2412
+        printf("    w(s) <= 3: estimating error vector as [s, 0(12)]\n");
+#endif
+        /* set e_hat = [s 0(12)] */
+        e_hat = (s << 12) & 0xfff000;
+    }
+    else {
+        /* step 3: search for p[i] s.t. w(s+p[i]) <= 2 */
+#if DEBUG_FEC_GOLAY2412
+        printf("    searching for w(s + p_i) <= 2...\n");
+#endif
+        int8_t s_index = golay2412_parity_search(s, _A);
+
+        if (s_index >= 0) {
+            /* vector found! */
+#if DEBUG_FEC_GOLAY2412
+            printf("    w(s + p[%2i]) <= 2: estimating error vector as [s+p[%2i],"
+                   "u[%2i]]\n", s_index, s_index, s_index);
+#endif
+            /* NOTE : uj = 1 << (12-j-1) */
+            e_hat = ((s ^ _A[s_index]) << 12) | (1 << (11 - s_index));
+        }
+        else {
+            /* step 4: compute s*P */
+            uint32_t sP = golay2412_matrix_mul(s, _A, 12);
+#if DEBUG_FEC_GOLAY2412
+            printf("s*P: "); liquid_print_bitstring(sP, 12); printf("\n");
+#endif
+
+            /* compute weight of sP (12 bits) */
+            uint8_t wsP = bitarithm_bits_set_u32(sP & 0x00000fff);
+#if DEBUG_FEC_GOLAY2412
+            printf("w(s*P) = %u\n", wsP);
+#endif
+
+            if (wsP == 2 || wsP == 3) {
+                /* step 5: set e = [0, s*P] */
+#if DEBUG_FEC_GOLAY2412
+                printf("    w(s*P) in [2,3]: estimating error vector as [0(12), s*P]\n");
+#endif
+                e_hat = sP;
+            }
+            else {
+                /* step 6: search for p[i] s.t. w(s*P + p[i]) == 2... */
+
+#if DEBUG_FEC_GOLAY2412
+                printf("    searching for w(s*P + p_i) == 2...\n");
+#endif
+                int8_t sP_index = golay2412_parity_search(sP, _A);
+
+                if (sP_index >= 0) {
+                    /* vector found! */
+#if DEBUG_FEC_GOLAY2412
+                    printf("    w(s*P + p[%2i]) == 2: estimating error vector as [u[%2i],"
+                           "s*P+p[%2i]]\n", sP_index, sP_index, sP_index);
+#endif
+                    /* NOTE : uj = 1 << (12-j-1)
+                     *      [      uj << 1 2    ] [    sP + p[j]    ] */
+                    e_hat = (1L << (23 - sP_index)) | (sP ^ _A[sP_index]);
+                }
+                else {
+                    /* step 7: decoding error */
+#if DEBUG_FEC_GOLAY2412
+                    printf("  **** decoding error\n");
+#endif
+                }
+            }
+        }
+    }
+
+    /* step 8: compute estimated transmitted message: v_hat = r + e_hat */
+    v_hat = _sym_enc ^ e_hat;
+#if DEBUG_FEC_GOLAY2412
+    printf("r (recevied vector):            ");
+    liquid_print_bitstring(_sym_enc, 24); printf("\n");
+    printf("e-hat (estimated error vector): ");
+    liquid_print_bitstring(e_hat, 24);    printf("\n");
+    printf("v-hat (estimated tx vector):    ");
+    liquid_print_bitstring(v_hat, 24);    printf("\n");
+#endif
+
+    /* compute estimated original message: (last 12 bits of encoded message) */
+    m_hat = v_hat & 0x0fff;
+
+    return m_hat;
+}
+
+void golay2412_encode(uint32_t _dec_msg_len,
+                      unsigned char *_msg_dec,
+                      unsigned char *_msg_enc)
+{
+    uint32_t i = 0;         /* decoded byte counter */
+    uint32_t j = 0;         /* encoded byte counter */
+    uint32_t m0;            /* first 12-bit symbol (uncoded) */
+    uint32_t v0;            /* first 24-bit symbol (encoded) */
+    unsigned char s0;       /* first 8-bit symbol */
+
+    /* determine remainder of input length / 3 */
+    uint32_t r = _dec_msg_len % 3;
+
+    for (i = 0; i < _dec_msg_len - r; i += 3) {
+        uint32_t m1;            /* second 12-bit symbol (uncoded) */
+        uint32_t v1;            /* second 24-bit symbol (encoded) */
+        unsigned char s1, s2;   /* second and third 8-bit symbols */
+
+        /* strip three input bytes (two uncoded symbols) */
+        s0 = _msg_dec[i + 0];
+        s1 = _msg_dec[i + 1];
+        s2 = _msg_dec[i + 2];
+
+        /* pack into two 12-bit symbols */
+        m0 = ((s0 << 4) & 0x0ff0) | ((s1 >> 4) & 0x000f);
+        m1 = ((s1 << 8) & 0x0f00) | ((s2) & 0x00ff);
+
+        /* encode each 12-bit symbol into a 24-bit symbol */
+        v0 = golay2412_encode_symbol(m0, golay2412_Gt);
+        v1 = golay2412_encode_symbol(m1, golay2412_Gt);
+
+        /* unpack two 24-bit symbols into six 8-bit bytes
+         * retaining order of bits in output */
+        _msg_enc[j + 0] = (v0 >> 16) & 0xff;
+        _msg_enc[j + 1] = (v0 >>  8) & 0xff;
+        _msg_enc[j + 2] = (v0) & 0xff;
+        _msg_enc[j + 3] = (v1 >> 16) & 0xff;
+        _msg_enc[j + 4] = (v1 >>  8) & 0xff;
+        _msg_enc[j + 5] = (v1) & 0xff;
+
+        j += 6;
+    }
+
+    /* if input length isn't divisible by 3, encode last 1 or two bytes */
+    for (i = _dec_msg_len - r; i < _dec_msg_len; i++) {
+        /* strip last input symbol */
+        s0 = _msg_dec[i];
+
+        /* extend as 12-bit symbol */
+        m0 = s0;
+
+        /* encode into 24-bit symbol */
+        v0 = golay2412_encode_symbol(m0, golay2412_Gt);
+
+        /* unpack one 24-bit symbol into three 8-bit bytes, and
+         * append to output array */
+        _msg_enc[j + 0] = (v0 >> 16) & 0xff;
+        _msg_enc[j + 1] = (v0 >>  8) & 0xff;
+        _msg_enc[j + 2] = (v0) & 0xff;
+
+        j += 3;
+    }
+
+    assert( j == block_get_enc_msg_len(_dec_msg_len, 12, 24));
+    assert( i == _dec_msg_len);
+}
+
+void golay2412_decode(uint32_t _dec_msg_len,
+                      unsigned char *_msg_enc,
+                      unsigned char *_msg_dec)
+{
+    uint32_t i = 0;                     /* decoded byte counter */
+    uint32_t j = 0;                     /* encoded byte counter */
+    uint32_t v0;                        /* first 24-bit encoded symbol */
+    uint32_t m0_hat;                    /* first 12-bit decoded symbol */
+    unsigned char r0, r1, r2;           /* first three 8-bit bytes */
+
+    /* determine remainder of input length / 3 */
+    uint32_t r = _dec_msg_len % 3;
+
+    for (i = 0; i < _dec_msg_len - r; i += 3) {
+        uint32_t v1;                        /* second 24-bit encoded symbol */
+        uint32_t m1_hat;                    /* second 12-bit decoded symbol */
+        unsigned char r3, r4, r5;           /* last three 8-bit bytes */
+
+        /* strip six input bytes (two encoded symbols) */
+        r0 = _msg_enc[j + 0];
+        r1 = _msg_enc[j + 1];
+        r2 = _msg_enc[j + 2];
+        r3 = _msg_enc[j + 3];
+        r4 = _msg_enc[j + 4];
+        r5 = _msg_enc[j + 5];
+
+        /* pack six 8-bit symbols into two 24-bit symbols */
+        v0 =  (((uint32_t)r0 << 16) & 0xff0000)
+             | (((uint32_t)r1 <<  8) & 0x00ff00)
+             | (((uint32_t)r2 <<  0) & 0x0000ff);
+
+        v1 =  (((uint32_t)r3 << 16) & 0xff0000)
+             | (((uint32_t)r4 <<  8) & 0x00ff00)
+             | (((uint32_t)r5 <<  0) & 0x0000ff);
+
+        /* decode each symbol into a 12-bit symbol */
+        m0_hat = golay2412_decode_symbol(v0, golay2412_P, golay2412_H);
+        m1_hat = golay2412_decode_symbol(v1, golay2412_P, golay2412_H);
+
+        /* unpack two 12-bit symbols into three 8-bit bytes */
+        _msg_dec[i + 0] = ((m0_hat >> 4) & 0xff);
+        _msg_dec[i + 1] = ((m0_hat << 4) & 0xf0) | ((m1_hat >> 8) & 0x0f);
+        _msg_dec[i + 2] = ((m1_hat) & 0xff);
+
+        j += 6;
+    }
+
+    /* if input length isn't divisible by 3, decode last 1 or two bytes */
+    for (i = _dec_msg_len - r; i < _dec_msg_len; i++) {
+        /* strip last input symbol (three bytes) */
+        r0 = _msg_enc[j + 0];
+        r1 = _msg_enc[j + 1];
+        r2 = _msg_enc[j + 2];
+
+        /* pack three 8-bit symbols into one 24-bit symbol */
+        v0 =  (((uint32_t)r0 << 16) & 0xff0000)
+             | (((uint32_t)r1 <<  8) & 0x00ff00)
+             | (((uint32_t)r2 <<  0) & 0x0000ff);
+
+        /* decode into a 12-bit symbol */
+        m0_hat = golay2412_decode_symbol(v0, golay2412_P, golay2412_H);
+
+        /* retain last 8 bits of 12-bit symbol */
+        _msg_dec[i] = m0_hat & 0xff;
+
+        j += 3;
+    }
+
+    assert( j == block_get_enc_msg_len(_dec_msg_len, 12, 24));
+    assert( i == _dec_msg_len);
+}
diff --git a/sys/include/ecc/golay2412.h b/sys/include/ecc/golay2412.h
new file mode 100644
index 0000000000000000000000000000000000000000..898999c31c3165c1ddcf4933b86d4be905efa37e
--- /dev/null
+++ b/sys/include/ecc/golay2412.h
@@ -0,0 +1,77 @@
+/*
+ * Copyright (c) 2007 - 2015 Joseph Gaeddert
+ * Copyright (c) 2018        HAW Hamburg
+ *
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
+ * of this software and associated documentation files (the "Software"), to deal
+ * in the Software without restriction, including without limitation the rights
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+ * copies of the Software, and to permit persons to whom the Software is
+ * furnished to do so, subject to the following conditions:
+ *
+ * The above copyright notice and this permission notice shall be included in
+ * all copies or substantial portions of the Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+ * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+ * THE SOFTWARE.
+ */
+
+/**
+ * @ingroup     sys_ecc
+ * @{
+ *
+ * @file
+ * @brief       Golay(24,12) half-rate forward error-correction code
+ *
+ * References:
+ *  [Lin:2004] Lin, Shu and Costello, Daniel L. Jr., "Error Control
+ *      Coding," Prentice Hall, New Jersey, 2nd edition, 2004.
+ *
+ * @author      Joseph Gaeddert
+ * @author      Peter Kietzmann <peter.kietzmann@haw.hamburg.de>
+ *
+ * @}
+ */
+
+#ifndef ECC_GOLAY2412_H
+#define ECC_GOLAY2412_H
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/**
+ * @brief encode block of data using Golay(24,12) encoder
+ *
+ * @pre _dec_msg_len and sizeof(_msg_dec) accordingly have to be multiples of 3
+ *
+ * @param[in] _dec_msg_len     decoded message length (number of bytes)
+ * @param[in] _msg_dec         decoded message [size: 1 x _dec_msg_len]
+ * @param[in] _msg_enc         encoded message [size: 1 x 2*_dec_msg_len]
+ */
+void golay2412_encode(uint32_t _dec_msg_len,
+                      unsigned char *_msg_dec,
+                      unsigned char *_msg_enc);
+
+/**
+ * @brief decode block of data using Golay(24,12) decoder
+ *
+ * @param[in] _dec_msg_len     decoded message length (number of bytes)
+ * @param[in] _msg_dec         encoded message [size: 1 x 2*_dec_msg_len]
+ * @param[in] _msg_enc         decoded message [size: 1 x _dec_msg_len]
+ */
+void golay2412_decode(uint32_t _dec_msg_len,
+                      unsigned char *_msg_enc,
+                      unsigned char *_msg_dec);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* ECC_GOLAY2412_H */
+/** @} */
diff --git a/tests/unittests/tests-ecc/Makefile.include b/tests/unittests/tests-ecc/Makefile.include
index 84b90addc6cb2472a3f77937c78d9cb629b2d13a..ed7dbc9eeb8d1989ea7baa51b5eebbac87b63488 100644
--- a/tests/unittests/tests-ecc/Makefile.include
+++ b/tests/unittests/tests-ecc/Makefile.include
@@ -1 +1,2 @@
+USEMODULE += ecc_golay2412
 USEMODULE += ecc_hamming256
diff --git a/tests/unittests/tests-ecc/tests-ecc.c b/tests/unittests/tests-ecc/tests-ecc.c
index 2a022adc8acdb5f895189bc8ca80ed07dc2b55f0..98f5d95b449a5e70dc4e74e8a169ad9aadfe7324 100644
--- a/tests/unittests/tests-ecc/tests-ecc.c
+++ b/tests/unittests/tests-ecc/tests-ecc.c
@@ -20,6 +20,18 @@
 #include "embUnit.h"
 
 #include "ecc/hamming256.h"
+#include "ecc/golay2412.h"
+
+/* source for random bytes: https://www.random.org/bytes */
+unsigned char data_in[] = { 201, 240, 154, 5, 227, 60, 116, 192, 214 };
+/* encoded sequence */
+unsigned char msg_enc[] = { 220, 124, 159,  19, 64, 154, 135, 208,  94,
+                            227,  35,  60, 248, 39,  76, 187,  16, 214 };
+/* Added errors. golay(24, 12) can correct up to 3 errors in one 24-bit
+ * symbol (= 3 bytes). Positions for bitflips generated at
+ * https://www.random.org/bytes */
+unsigned char msg_enc_3_err_per_symbol[] = { 252, 60, 191,  18, 64, 190, 135, 209,  22,
+                                             227, 43, 188, 252, 33,  76,  57,  16, 212 };
 
 static void test_hamming256_single(void)
 {
@@ -75,11 +87,56 @@ static void test_hamming256_padding(void)
     TEST_ASSERT_EQUAL_INT(Hamming_ERROR_ECC, result);
 }
 
+static void test_golay2412_message_encode(void)
+{
+    unsigned char encoded[2 * sizeof(data_in)];
+
+    golay2412_encode(sizeof(data_in), &data_in[0], &encoded[0]);
+
+    TEST_ASSERT_EQUAL_INT(0, memcmp(&msg_enc, &encoded, sizeof(encoded)));
+}
+
+static void test_golay2412_message_noerr(void)
+{
+    unsigned char result[sizeof(data_in)];
+    unsigned char encoded[2 * sizeof(data_in)];
+
+    golay2412_encode(sizeof(data_in), &data_in[0], &encoded[0]);
+    golay2412_decode(sizeof(result), &encoded[0], &result[0]);
+    TEST_ASSERT_EQUAL_INT(0, memcmp(&data_in, &result, sizeof(data_in)));
+}
+
+static void test_golay2412_message_decode_success(void)
+{
+    unsigned char result[sizeof(data_in)];
+
+    golay2412_decode(sizeof(result), &msg_enc_3_err_per_symbol[0], &result[0]);
+
+    TEST_ASSERT_EQUAL_INT(0, memcmp(&data_in, &result, sizeof(data_in)));
+}
+
+static void test_golay2412_message_decode_fail(void)
+{
+    unsigned char result[sizeof(data_in)];
+
+    /* adding a 4th error to the first symbol leads to an
+       uncorrectable sequence */
+    msg_enc_3_err_per_symbol[0] ^= (1L << 6);
+
+    golay2412_decode(sizeof(result), &msg_enc_3_err_per_symbol[0], &result[0]);
+
+    TEST_ASSERT(memcmp(&data_in, &result, sizeof(data_in)));
+}
+
 TestRef test_all(void)
 {
     EMB_UNIT_TESTFIXTURES(fixtures) {
         new_TestFixture(test_hamming256_single),
         new_TestFixture(test_hamming256_padding),
+        new_TestFixture(test_golay2412_message_encode),
+        new_TestFixture(test_golay2412_message_noerr),
+        new_TestFixture(test_golay2412_message_decode_success),
+        new_TestFixture(test_golay2412_message_decode_fail),
     };
 
     EMB_UNIT_TESTCALLER(EccTest, NULL, NULL, fixtures);