diff --git a/Makefile.dep b/Makefile.dep
index d2eb6eb02ac7796e591671539bca98f312f5ab9c..39bd45a516031f45a7045271b60e32e7041996bc 100644
--- a/Makefile.dep
+++ b/Makefile.dep
@@ -301,6 +301,10 @@ ifneq (,$(filter posix,$(USEMODULE)))
   USEMODULE += vtimer
 endif
 
+ifneq (,$(filter sem,$(USEMODULE)))
+  USEMODULE += vtimer
+endif
+
 ifneq (,$(filter vtimer,$(USEMODULE)))
   USEMODULE += xtimer
   USEMODULE += timex
diff --git a/sys/Makefile b/sys/Makefile
index efb6384dadf17aa24bb879fa44753af6ab7461e1..78f375543289937500880f0dc7e2cb4d09747eae 100644
--- a/sys/Makefile
+++ b/sys/Makefile
@@ -68,6 +68,9 @@ endif
 ifneq (,$(filter netopt,$(USEMODULE)))
     DIRS += net/crosslayer/netopt
 endif
+ifneq (,$(filter sem,$(USEMODULE)))
+    DIRS += sem
+endif
 
 DIRS += $(dir $(wildcard $(addsuffix /Makefile, ${USEMODULE})))
 
diff --git a/sys/include/sem.h b/sys/include/sem.h
new file mode 100644
index 0000000000000000000000000000000000000000..6d647e4dcda09e4612879b1be203c959ce59b1d5
--- /dev/null
+++ b/sys/include/sem.h
@@ -0,0 +1,145 @@
+/*
+ * Copyright (C) 2015 Martine Lenders <mlenders@inf.fu-berlin.de>
+ *
+ * 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_sem Semaphores
+ * @ingroup     sys
+ * @brief       Lightweight semaphore implementation
+ * @{
+ *
+ * @file
+ * @brief   Semaphore definitions
+ *
+ * @author  Martine Lenders <mlenders@inf.fu-berlin.de>
+ * @author  Christian Mehlis <mehlis@inf.fu-berlin.de>
+ * @author  René Kijewski <kijewski@inf.fu-berlin.de>
+ */
+#ifndef SEM_H_
+#define SEM_H_
+
+#include "priority_queue.h"
+#include "timex.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/**
+ * @brief A Semaphore.
+ */
+typedef struct {
+    volatile unsigned int value;    /**< value of the semaphore */
+    priority_queue_t queue;         /**< list of threads waiting for the semaphore */
+} sem_t;
+
+/**
+ * @brief   Creates semaphore.
+ *
+ * @see <a href="http://pubs.opengroup.org/onlinepubs/9699919799/functions/sem_init.html">
+ *          The Open Group Base Specifications Issue 7, sem_init()
+ *      </a> (without `pshared` parameter)
+ *
+ * @param[out] sem  The created semaphore.
+ * @param[in] value Initial value for the semaphore.
+ *
+ * @return  0 on success.
+ * @return  -EINVAL, if semaphore is invalid.
+ */
+int sem_create(sem_t *sem, unsigned int value);
+
+/**
+ * @brief   Destroys a semaphore.
+ *
+ * @see <a href="http://pubs.opengroup.org/onlinepubs/9699919799/functions/sem_destroy.html">
+ *          The Open Group Base Specifications Issue 7, sem_destroy()
+ *      </a>
+ *
+ * @param[in] sem   The semaphore to destroy.
+ *
+ * @return  0 on success.
+ * @return  -EINVAL, if semaphore is invalid.
+ */
+int sem_destroy(sem_t *sem);
+
+/**
+ * @brief   Wait for a semaphore being posted.
+ *
+ * @pre Message queue of active thread is initialized (see @ref msg_init_queue()).
+ *
+ * @param[in]  sem      A semaphore.
+ * @param[in]  timeout  Time until the semaphore times out. NULL for no timeout.
+ * @param[out] msg      Container for a spurious message during the timed wait (result == -EAGAIN).
+ *
+ * @return  0 on success
+ * @return  -EINVAL, if semaphore is invalid.
+ * @return  -ETIMEDOUT, if the semaphore times out.
+ * @return  -ECANCELED, if the semaphore was destroyed.
+ * @return  -EAGAIN, if the thread received a message while waiting for the lock.
+ */
+int sem_wait_timed_msg(sem_t *sem, timex_t *timeout, msg_t *msg);
+
+/**
+ * @brief   Wait for a semaphore being posted (without timeout).
+ *
+ * @param[in]  sem  A semaphore.
+ * @param[out] msg  Container for a spurious message during the timed wait (result == -EAGAIN).
+ *
+ * @return  0 on success
+ * @return  -EINVAL, if semaphore is invalid.
+ * @return  -ECANCELED, if the semaphore was destroyed.
+ * @return  -EAGAIN, if the thread received a message while waiting for the lock.
+ */
+static inline int sem_wait_msg(sem_t *sem, msg_t *msg)
+{
+    return sem_wait_timed_msg(sem, NULL, msg);
+}
+
+/**
+ * @brief   Wait for a semaphore being posted (dropping spurious messages).
+ * @details Any spurious messages received while waiting for the semaphore are silently dropped.
+ *
+ * @param[in]  sem      A semaphore.
+ * @param[in]  timeout  Time until the semaphore times out. NULL for no timeout.
+ *
+ * @return  0 on success
+ * @return  -EINVAL, if semaphore is invalid.
+ * @return  -ETIMEDOUT, if the semaphore times out.
+ * @return  -ECANCELED, if the semaphore was destroyed.
+ */
+int sem_wait_timed(sem_t *sem, timex_t *timeout);
+
+/**
+ * @brief   Wait for a semaphore being posted (without timeout, dropping spurious messages).
+ *
+ * @param[in]  sem  A semaphore.
+ *
+ * @return  0 on success
+ * @return  -EINVAL, if semaphore is invalid.
+ * @return  -ECANCELED, if the semaphore was destroyed.
+ */
+static inline int sem_wait(sem_t *sem)
+{
+    return sem_wait_timed(sem, NULL);
+}
+
+/**
+ * @brief   Signal semaphore.
+ *
+ * @param[in] sem   A semaphore.
+ *
+ * @return  -EINVAL, if semaphore is invalid.
+ * @return  -EOVERFLOW, if the semaphore's value would overflow.
+ */
+int sem_post(sem_t *sem);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* SEM_H_ */
+/** @} */
diff --git a/sys/sem/Makefile b/sys/sem/Makefile
new file mode 100644
index 0000000000000000000000000000000000000000..48422e909a47d7cd428d10fa73825060ccc8d8c2
--- /dev/null
+++ b/sys/sem/Makefile
@@ -0,0 +1 @@
+include $(RIOTBASE)/Makefile.base
diff --git a/sys/sem/sem.c b/sys/sem/sem.c
new file mode 100644
index 0000000000000000000000000000000000000000..a4256069b760a386740ae139dff1919425663364
--- /dev/null
+++ b/sys/sem/sem.c
@@ -0,0 +1,173 @@
+/*
+ * Copyright (C) 2013-15 Freie Universität Berlin
+ *
+ * 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.
+ */
+
+/**
+ * @{
+ *
+ * @file
+ *
+ * @author  Christian Mehlis <mehlis@inf.fu-berlin.de>
+ * @author  Martine Lenders <mlenders@inf.fu-berlin.de>
+ * @author  René Kijewski <kijewski@inf.fu-berlin.de>
+ */
+
+#include <assert.h>
+#include <errno.h>
+#include <limits.h>
+
+#include "irq.h"
+#include "msg.h"
+#include "vtimer.h"
+
+#include "sem.h"
+
+#define ENABLE_DEBUG (0)
+#include "debug.h"
+
+#define MSG_SIGNAL         (0x0501)
+#define MSG_TIMEOUT        (0x0502)
+#define MSG_DESTROYED      (0x0503)
+
+int sem_create(sem_t *sem, unsigned int value)
+{
+    if (sem == NULL) {
+#ifdef MODULE_POSIX_SEMAPHORE
+        errno = EINVAL;
+#endif
+        return -EINVAL;
+    }
+    sem->value = value;
+    /* waiters for the mutex */
+    sem->queue.first = NULL;
+    return 0;
+}
+
+int sem_destroy(sem_t *sem)
+{
+    unsigned int old_state;
+    priority_queue_node_t *next;
+    if (sem == NULL) {
+#ifdef MODULE_POSIX_SEMAPHORE
+        errno = EINVAL;
+#endif
+        return -EINVAL;
+    }
+    old_state = disableIRQ();
+    while ((next = priority_queue_remove_head(&sem->queue)) != NULL) {
+        msg_t msg;
+        kernel_pid_t pid = (kernel_pid_t)next->data;
+        msg.type = MSG_DESTROYED;
+        msg.content.ptr = (void *) sem;
+        msg_send_int(&msg, pid);
+    }
+    restoreIRQ(old_state);
+    return 0;
+}
+
+int sem_wait_timed_msg(sem_t *sem, timex_t *timeout, msg_t *msg)
+{
+    if (sem == NULL) {
+        return -EINVAL;
+    }
+    while (1) {
+        unsigned old_state = disableIRQ();
+        priority_queue_node_t n;
+        vtimer_t timeout_timer;
+
+        unsigned value = sem->value;
+        if (value != 0) {
+            sem->value = value - 1;
+            restoreIRQ(old_state);
+            return 0;
+        }
+
+        /* I'm going blocked */
+        n.priority = (uint32_t)sched_active_thread->priority;
+        n.data = (unsigned int)sched_active_pid;
+        n.next = NULL;
+        priority_queue_add(&sem->queue, &n);
+
+        DEBUG("sem_wait: %" PRIkernel_pid ": Adding node to semaphore queue: prio: %" PRIu32 "\n",
+              sched_active_thread->pid, sched_active_thread->priority);
+
+        if (timeout != NULL) {
+            vtimer_set_msg(&timeout_timer, *timeout, sched_active_pid,
+                           MSG_TIMEOUT, sem);
+        }
+
+        restoreIRQ(old_state);
+        msg_receive(msg);
+
+        if (timeout != NULL) {
+            vtimer_remove(&timeout_timer);  /* remove timer just to be sure */
+        }
+
+        if (msg->content.ptr != (void *) sem) {
+            return -EAGAIN;
+        }
+
+        switch (msg->type) {
+            case MSG_SIGNAL:
+                continue;
+            case MSG_TIMEOUT:
+                return -ETIMEDOUT;
+            case MSG_DESTROYED:
+                return -ECANCELED;
+            default:
+                return -EAGAIN;
+        }
+    }
+}
+
+int sem_wait_timed(sem_t *sem, timex_t *timeout)
+{
+    int result;
+    do {
+        msg_t msg;
+        result = sem_wait_timed_msg(sem, timeout, &msg);
+        DEBUG("sem_wait: %" PRIkernel_pid ": Discarding message from %" PRIkernel_pid "\n",
+              sched_active_thread->pid, msg->sender_pid);
+    } while (result == -EAGAIN);
+    return result;
+}
+
+int sem_post(sem_t *sem)
+{
+    unsigned int old_state, value;
+    priority_queue_node_t *next;
+    if (sem == NULL) {
+        return -EINVAL;
+    }
+    old_state = disableIRQ();
+    value = sem->value;
+    if (value == UINT_MAX) {
+        restoreIRQ(old_state);
+        return -EOVERFLOW;
+    }
+    ++sem->value;
+    next = priority_queue_remove_head(&sem->queue);
+    if (next) {
+        uint16_t prio = (uint16_t)next->priority;
+        kernel_pid_t pid = (kernel_pid_t) next->data;
+        msg_t msg;
+        DEBUG("sem_post: %" PRIkernel_pid ": waking up %" PRIkernel_pid "\n",
+              sched_active_thread->pid, next_process->pid);
+        msg.type = MSG_SIGNAL;
+        msg.content.ptr = (void *) sem;
+        msg_send_int(&msg, pid);
+        restoreIRQ(old_state);
+        sched_switch(prio);
+    }
+    else {
+        restoreIRQ(old_state);
+    }
+
+    return 1;
+}
+
+/** @} */