diff --git a/sys/posix/pthread/include/pthread.h b/sys/posix/pthread/include/pthread.h
index acd5e8e3060497fb418ceca8d490b5ed0e8e4dd5..d0949e417fe5f0d4c49dfbb47b8d65eab36a71c3 100644
--- a/sys/posix/pthread/include/pthread.h
+++ b/sys/posix/pthread/include/pthread.h
@@ -15,6 +15,7 @@
 #include "pthread_threading.h"
 #include "pthread_mutex_attr.h"
 #include "pthread_mutex.h"
+#include "pthread_rwlock_attr.h"
 #include "pthread_rwlock.h"
 #include "pthread_spin.h"
 #include "pthread_barrier.h"
diff --git a/sys/posix/pthread/include/pthread_rwlock.h b/sys/posix/pthread/include/pthread_rwlock.h
index dd631575a94e2e3891ca2325b10ec51269665ef6..b5b5b772193fec3b753c03e7dee9cc1163fc2795 100644
--- a/sys/posix/pthread/include/pthread_rwlock.h
+++ b/sys/posix/pthread/include/pthread_rwlock.h
@@ -2,57 +2,154 @@
  * @ingroup pthread
  */
 
-typedef unsigned long int pthread_rwlock_t;
-typedef unsigned long int pthread_rwlockattr_t;
+#include "queue.h"
+#include "tcb.h"
 
-/* Initialize read-write lock RWLOCK using attributes ATTR, or use
- the default values if later is NULL.  */
-int pthread_rwlock_init(pthread_rwlock_t *rwlock,
-        const pthread_rwlockattr_t *attr);
+#include <errno.h>
+#include <stdbool.h>
 
-/* Destroy read-write lock RWLOCK.  */
+/**
+ * @brief     A fair reader writer lock.
+ * @details   The implementation ensures that readers and writers of the same priority
+ *            won't starve each other.
+ *            E.g. no new readers will get into the critical section
+ *            if a writer of the same or a higher priority already waits for the lock.
+ */
+typedef struct pthread_rwlock
+{
+    /**
+     * @brief     The current amount of reader inside the critical section.
+     * @details
+     *            * `== 0`: no thread is in the critical section.
+     *            * `> 0`: the number of readers currently in the critical section.
+     *            * `< 0`: a writer is currently in the critical section.
+     */
+    int readers;
+
+    /**
+     * @brief     Queue of waiting threads.
+     */
+    queue_node_t queue;
+
+    /**
+     * @brief     Provides mutual exclusion on reading and writing on the structure.
+     */
+    mutex_t mutex;
+} pthread_rwlock_t;
+
+/**
+ * @brief     Internal structure that stores one waiting thread.
+ */
+typedef struct __pthread_rwlock_waiter_node
+{
+    bool is_writer; /**< `false`: reader; `true`: writer */
+    tcb_t *thread; /**< waiting thread */
+    queue_node_t qnode; /**< Node to store in `pthread_rwlock_t::queue`. */
+    bool continue_; /**< This is not a spurious wakeup. */
+} __pthread_rwlock_waiter_node_t;
+
+/**
+ * @brief           Initialize a reader/writer lock.
+ * @details         A zeroed out datum is initialized.
+ * @param[in,out]   rwlock   Lock to initialize.
+ * @param[in]       attr     Unused.
+ * @returns         `0` on success.
+ *                  `EINVAL` if `rwlock == NULL`.
+ */
+int pthread_rwlock_init(pthread_rwlock_t *rwlock, const pthread_rwlockattr_t *attr);
+
+/**
+ * @brief           Destroy a reader/writer lock.
+ * @details         This is a no-op.
+ *                  Destroying a reader/writer lock while a thread holds it, or
+ *                  there are threads waiting for it causes undefined behavior.
+ *                  Datum must be reinitialized before using it again.
+ * @param[in]       rwlock   Lock to destroy.
+ * @returns         `0` on success.
+ *                  `EINVAL` if `rwlock == NULL`.
+ *                  `EBUSY` if the lock was in use.
+ */
 int pthread_rwlock_destroy(pthread_rwlock_t *rwlock);
 
-/* Acquire read lock for RWLOCK.  */
+/**
+ * @brief           Lock a reader/writer lock for reading.
+ * @details         This function may block.
+ * @param[in]       rwlock   Lock to acquire for reading.
+ * @returns         `0` if the lock could be acquired.
+ *                  `EINVAL` if `rwlock == NULL`.
+ */
 int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock);
 
-/* Try to acquire read lock for RWLOCK.  */
+/**
+ * @brief           Try to lock a reader/writer lock for reader.
+ * @details         This function won't block.
+ * @param[in]       rwlock   Lock to acquire for reading.
+ * @returns         `0` if the lock could be acquired.
+ *                  `EBUSY` if acquiring the lock cannot be done without blocking.
+ *                  `EINVAL` if `rwlock == NULL`.
+ */
 int pthread_rwlock_tryrdlock(pthread_rwlock_t *rwlock);
 
-/* Try to acquire read lock for RWLOCK or return after specfied time.  */
-int pthread_rwlock_timedrdlock(pthread_rwlock_t *rwlock,
-        const struct timespec *abstime);
+/**
+ * @brief           Try to acquire a read lock in a given timeframe
+ * @param[in]       rwlock    Lock to acquire for reading.
+ * @param[in]       abstime   Maximum timestamp when to wakeup, absolute.
+ * @returns         `0` if the lock could be acquired.
+ *                  `EDEADLK` if the lock could not be acquired in the given timeframe.
+ *                  `EINVAL` if `rwlock == NULL`.
+ */
+int pthread_rwlock_timedrdlock(pthread_rwlock_t *rwlock, const struct timespec *abstime);
 
-/* Acquire write lock for RWLOCK.  */
+/**
+ * @brief           Lock a reader/writer lock for writing.
+ * @details         This function may block.
+ * @param[in]       rwlock   Lock to acquire for writing.
+ * @returns         `0` if the lock could be acquired.
+ *                  `EINVAL` if `rwlock == NULL`.
+ */
 int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock);
 
-/* Try to acquire write lock for RWLOCK.  */
+/**
+ * @brief           Try to lock a reader/writer lock for writing.
+ * @details         This function won't block.
+ * @param[in]       rwlock   Lock to acquire for writing.
+ * @returns         `0` if the lock could be acquired.
+ *                  `EBUSY` if acquiring the lock cannot be done without blocking.
+ *                  `EINVAL` if `rwlock == NULL`.
+ */
 int pthread_rwlock_trywrlock(pthread_rwlock_t *rwlock);
 
-/* Try to acquire write lock for RWLOCK or return after specfied time.  */
-int pthread_rwlock_timedwrlock(pthread_rwlock_t *rwlock,
-        const struct timespec *abstime);
+/**
+ * @brief           Try to acquire a write lock in a given timeframe
+ * @param[in]       rwlock    Lock to acquire for writing.
+ * @param[in]       abstime   Maximum timestamp when to wakeup, absolute.
+ * @returns         `0` if the lock could be acquired.
+ *                  `EDEADLK` if the lock could not be acquired in the given timeframe.
+ *                  `EINVAL` if `rwlock == NULL`.
+ */
+int pthread_rwlock_timedwrlock(pthread_rwlock_t *rwlock, const struct timespec *abstime);
 
-/* Unlock RWLOCK.  */
+/**
+ * @brief           Unlock the reader/writer lock.
+ * @details         Must only be used if the lock is currently held.
+ *                  You may release the lock out of another thread, but the *lock*, *operate*, *unlock* logic must not be broken.
+ * @param[in]       rwlock   Lock to release
+ * @returns         `0` on success.
+ *                  `EPERM` if the lock was not held for any operation.
+ *                  `EINVAL` if `rwlock == NULL`.
+ */
 int pthread_rwlock_unlock(pthread_rwlock_t *rwlock);
 
-/* Functions for handling read-write lock attributes.  */
-
-/* Initialize attribute object ATTR with default values.  */
-int pthread_rwlockattr_init(pthread_rwlockattr_t *attr);
-
-/* Destroy attribute object ATTR.  */
-int pthread_rwlockattr_destroy(pthread_rwlockattr_t *attr);
-
-/* Return current setting of process-shared attribute of ATTR in PSHARED.  */
-int pthread_rwlockattr_getpshared(const pthread_rwlockattr_t *attr,
-        int *pshared);
-
-/* Set process-shared attribute of ATTR to PSHARED.  */
-int pthread_rwlockattr_setpshared(pthread_rwlockattr_t *attr, int pshared);
-
-/* Return current setting of reader/writer preference.  */
-int pthread_rwlockattr_getkind_np(const pthread_rwlockattr_t *attr, int *pref);
+/**
+ * @brief            Internal function to determine of the lock can be acquired for reading.
+ * @param[in]        rwlock   Lock to query.
+ * @returns          `false` if locking for reading is possible without blocking.
+ */
+bool __pthread_rwlock_blocked_readingly(const pthread_rwlock_t *rwlock);
 
-/* Set reader/write preference.  */
-int pthread_rwlockattr_setkind_np(pthread_rwlockattr_t *attr, int pref);
+/**
+ * @brief            Internal function to determine of the lock can be acquired for writing.
+ * @param[in]        rwlock   Lock to query.
+ * @returns          `false` if locking for writing is possible without blocking.
+ */
+bool __pthread_rwlock_blocked_writingly(const pthread_rwlock_t *rwlock);
diff --git a/sys/posix/pthread/include/pthread_rwlock_attr.h b/sys/posix/pthread/include/pthread_rwlock_attr.h
new file mode 100644
index 0000000000000000000000000000000000000000..3e895b6c8929a731ea028513ae9fd613d51531a7
--- /dev/null
+++ b/sys/posix/pthread/include/pthread_rwlock_attr.h
@@ -0,0 +1,58 @@
+/**
+ * @ingroup pthread
+ */
+
+#include <errno.h>
+
+/**
+ * @brief     Attributes for a new reader/writer lock.
+ * @details   The options set in this struct will be ignored by pthread_rwlock_init().
+ */
+typedef struct pthread_rwlockattr
+{
+    /**
+     * @brief     Whether to share lock with child processes.
+     * @details   Valid values are `PTHREAD_PROCESS_SHARED` and `PTHREAD_PROCESS_PRIVATE`.
+     *            Since RIOT is a single-process operating system, this value is ignored.
+     */
+    int pshared;
+} pthread_rwlockattr_t;
+
+/**
+ * @brief           Initilize the attribute set with the defaults.
+ * @details         Default value for pshared: `PTHREAD_PROCESS_PRIVATE`.
+ *                  A zeroed out datum is initialized.
+ * @param[in,out]   attr   Attribute set to initialize.
+ * @returns         `0` on success.
+ *                  `EINVAL` if `attr == NULL`.
+ */
+int pthread_rwlockattr_init(pthread_rwlockattr_t *attr);
+
+/**
+ * @brief           Destroy an attribute set.
+ * @details         This function does nothing, don't bother calling it.
+ * @param[in,out]   attr   Attribute set to destroy.
+ * @returns         `0` on success.
+ *                  `EINVAL` if `attr == NULL`.
+ */
+int pthread_rwlockattr_destroy(pthread_rwlockattr_t *attr);
+
+/**
+ * @brief           Read whether to share the lock with child processes.
+ * @details         There are not child processes in RIOT.
+ * @param[in]       attr      Attribute set to query.
+ * @param[out]      pshared   Either `PTHREAD_PROCESS_SHARED` or `PTHREAD_PROCESS_PRIVATE`.
+ * @returns         `0` on success.
+ *                  `EINVAL` if `attr == NULL`.
+ */
+int pthread_rwlockattr_getpshared(const pthread_rwlockattr_t *attr, int *pshared);
+
+/**
+ * @brief           Set whether to share the lock with child processes.
+ * @details         There are not child processes in RIOT.
+ * @param[in,out]   attr      Attribute set to operate on.
+ * @param[in]       pshared   Either `PTHREAD_PROCESS_SHARED` or `PTHREAD_PROCESS_PRIVATE`.
+ * @returns         `0` on success.
+ *                  `EINVAL` if `attr == NULL` or a wrong value for `pshared` was supplied.
+ */
+int pthread_rwlockattr_setpshared(pthread_rwlockattr_t *attr, int pshared);
diff --git a/sys/posix/pthread/pthread_rwlock.c b/sys/posix/pthread/pthread_rwlock.c
new file mode 100644
index 0000000000000000000000000000000000000000..5b6a6b5af6c134964ca103bc3d2a75fc6d1a4c0d
--- /dev/null
+++ b/sys/posix/pthread/pthread_rwlock.c
@@ -0,0 +1,290 @@
+/*
+ * Copyright (C) 2014  René Kijewski  <rene.kijewski@fu-berlin.de>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
+ */
+
+/**
+ * @ingroup     pthread
+ * @{
+ *
+ * @file
+ * @brief       Implementation of a fair, POSIX conforming reader/writer lock.
+ *
+ * @author      René Kijewski <rene.kijewski@fu-berlin.de>
+ *
+ * @}
+ */
+
+#include "pthread.h"
+#include "sched.h"
+#include "vtimer.h"
+
+#include <stdint.h>
+#include <string.h>
+
+int pthread_rwlock_init(pthread_rwlock_t *rwlock, const pthread_rwlockattr_t *attr)
+{
+    (void) attr;
+
+    if (rwlock == NULL) {
+        return EINVAL;
+    }
+
+    memset(rwlock, 0, sizeof (*rwlock));
+    return 0;
+}
+
+int pthread_rwlock_destroy(pthread_rwlock_t *rwlock)
+{
+    if (rwlock == NULL) {
+        return EINVAL;
+    }
+
+    /* do not unlock the mutex, no need */
+    if ((mutex_trylock(&rwlock->mutex) == 0) || (rwlock->readers != 0)) {
+        return EBUSY;
+    }
+
+    return 0;
+}
+
+bool __pthread_rwlock_blocked_readingly(const pthread_rwlock_t *rwlock)
+{
+    if (rwlock->readers < 0) {
+        /* a writer holds the lock */
+        return true;
+    }
+
+    /* Determine if there is a writer waiting to get this lock who has a higher or the same priority: */
+
+    if (rwlock->queue.next == NULL) {
+        /* no waiting thread */
+        return false;
+    }
+
+    queue_node_t *qnode = rwlock->queue.next;
+    if (qnode->priority > active_thread->priority) {
+        /* the waiting thread has a lower priority */
+        return false;
+    }
+
+    /* if the waiting node is a writer, then we cannot enter the critical section (to prevent starving the writer) */
+    __pthread_rwlock_waiter_node_t *waiting_node = (__pthread_rwlock_waiter_node_t *) qnode->data;
+    return waiting_node->is_writer;
+}
+
+bool __pthread_rwlock_blocked_writingly(const pthread_rwlock_t *rwlock)
+{
+    /* if any thread holds the lock, then no writer may enter the critical section */
+    return rwlock->readers != 0;
+}
+
+static int pthread_rwlock_lock(pthread_rwlock_t *rwlock,
+                               bool (*is_blocked)(const pthread_rwlock_t *rwlock),
+                               bool is_writer,
+                               int incr_when_held,
+                               bool allow_spurious)
+{
+    if (rwlock == NULL) {
+        return EINVAL;
+    }
+
+    mutex_lock(&rwlock->mutex);
+    if (!is_blocked(rwlock)) {
+        rwlock->readers += incr_when_held;
+    }
+    else {
+        /* queue for the lock */
+        __pthread_rwlock_waiter_node_t waiting_node = {
+            .is_writer = is_writer,
+            .thread = (tcb_t *) active_thread,
+            .qnode = {
+                .next = NULL,
+                .data = (uintptr_t) &waiting_node,
+                .priority = active_thread->priority,
+            },
+            .continue_ = false,
+        };
+        queue_priority_add(&rwlock->queue, &waiting_node.qnode);
+
+        while (1) {
+            /* wait to be unlocked, so this thread can try to acquire the lock again */
+            mutex_unlock_and_sleep(&rwlock->mutex);
+
+            mutex_lock(&rwlock->mutex);
+            if (waiting_node.continue_) {
+                /* pthread_rwlock_unlock() already set rwlock->readers */
+                break;
+            }
+            else if (allow_spurious) {
+                queue_remove(&rwlock->queue, &waiting_node.qnode);
+                mutex_unlock(&rwlock->mutex);
+                return ETIMEDOUT;
+            }
+        }
+    }
+    mutex_unlock(&rwlock->mutex);
+
+    return 0;
+}
+
+static int pthread_rwlock_trylock(pthread_rwlock_t *rwlock,
+                                  bool (*is_blocked)(const pthread_rwlock_t *rwlock),
+                                  int incr_when_held)
+{
+    if (rwlock == NULL) {
+        return EINVAL;
+    }
+    else if (mutex_trylock(&rwlock->mutex) == 0) {
+        return EBUSY;
+    }
+    else if (is_blocked(rwlock)) {
+        mutex_unlock(&rwlock->mutex);
+        return EBUSY;
+    }
+
+    rwlock->readers += incr_when_held;
+
+    mutex_unlock(&rwlock->mutex);
+    return 0;
+}
+
+static int pthread_rwlock_timedlock(pthread_rwlock_t *rwlock,
+                                    bool (*is_blocked)(const pthread_rwlock_t *rwlock),
+                                    bool is_writer,
+                                    int incr_when_held,
+                                    const struct timespec *abstime)
+{
+    timex_t now, then;
+
+    then.seconds = abstime->tv_sec;
+    then.microseconds = abstime->tv_nsec / 1000u;
+    timex_normalize(&then);
+
+    vtimer_now(&now);
+
+    if (timex_cmp(then, now) <= 0) {
+        return ETIMEDOUT;
+    }
+    else {
+        timex_t reltime = timex_sub(then, now);
+
+        vtimer_t timer;
+        vtimer_set_wakeup(&timer, reltime, active_thread->pid);
+        int result = pthread_rwlock_lock(rwlock, is_blocked, is_writer, incr_when_held, true);
+        if (result != ETIMEDOUT) {
+            vtimer_remove(&timer);
+        }
+
+        return result;
+    }
+}
+
+int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock)
+{
+    return pthread_rwlock_lock(rwlock, __pthread_rwlock_blocked_readingly, false, +1, false);
+}
+
+int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock)
+{
+    return pthread_rwlock_lock(rwlock, __pthread_rwlock_blocked_writingly, true, -1, false);
+}
+
+int pthread_rwlock_tryrdlock(pthread_rwlock_t *rwlock)
+{
+    return pthread_rwlock_trylock(rwlock, __pthread_rwlock_blocked_readingly, +1);
+}
+
+int pthread_rwlock_trywrlock(pthread_rwlock_t *rwlock)
+{
+    return pthread_rwlock_trylock(rwlock, __pthread_rwlock_blocked_writingly, -1);
+}
+
+int pthread_rwlock_timedrdlock(pthread_rwlock_t *rwlock, const struct timespec *abstime)
+{
+    return pthread_rwlock_timedlock(rwlock, __pthread_rwlock_blocked_readingly, false, +1, abstime);
+}
+
+int pthread_rwlock_timedwrlock(pthread_rwlock_t *rwlock, const struct timespec *abstime)
+{
+    return pthread_rwlock_timedlock(rwlock, __pthread_rwlock_blocked_writingly, true, -1, abstime);
+}
+
+int pthread_rwlock_unlock(pthread_rwlock_t *rwlock)
+{
+    if (rwlock == NULL) {
+        return EINVAL;
+    }
+
+    mutex_lock(&rwlock->mutex);
+    if (rwlock->readers == 0) {
+        /* the lock is open */
+        mutex_unlock(&rwlock->mutex);
+        return EPERM;
+    }
+
+    if (rwlock->readers > 0) {
+        --rwlock->readers;
+    }
+    else {
+        rwlock->readers = 0;
+    }
+
+    if (rwlock->readers != 0 || rwlock->queue.next == NULL) {
+        /* this thread was not the last reader, or no one is waiting to aquire the lock */
+        mutex_unlock(&rwlock->mutex);
+        return 0;
+    }
+
+    /* wake up the next thread */
+    queue_node_t *qnode = queue_remove_head(&rwlock->queue);
+    __pthread_rwlock_waiter_node_t *waiting_node = (__pthread_rwlock_waiter_node_t *) qnode->data;
+    waiting_node->continue_ = true;
+    uint16_t prio = qnode->priority;
+    sched_set_status(waiting_node->thread, STATUS_PENDING);
+
+    if (waiting_node->is_writer) {
+        --rwlock->readers;
+    }
+    else {
+        ++rwlock->readers;
+
+        /* wake up further readers */
+        while (rwlock->queue.next) {
+            waiting_node = (__pthread_rwlock_waiter_node_t *) rwlock->queue.next->data;
+            if (waiting_node->is_writer) {
+                /* Not to be unfair to writers, we don't try to wake up readers that came after the first writer. */
+                break;
+            }
+            waiting_node->continue_ = true;
+
+            /* wake up this reader */
+            qnode = queue_remove_head(&rwlock->queue);
+            if (qnode->priority < prio) {
+                prio = qnode->priority;
+            }
+            sched_set_status(waiting_node->thread, STATUS_PENDING);
+
+            ++rwlock->readers;
+        }
+    }
+
+    mutex_unlock(&rwlock->mutex);
+
+    /* yield if a woken up thread had a higher priority */
+    sched_switch(active_thread->priority, prio);
+    return 0;
+}
diff --git a/sys/posix/pthread/pthread_rwlock_attr.c b/sys/posix/pthread/pthread_rwlock_attr.c
new file mode 100644
index 0000000000000000000000000000000000000000..cd73ef6635889f0f05f9ab73fe2af4b1d91d305a
--- /dev/null
+++ b/sys/posix/pthread/pthread_rwlock_attr.c
@@ -0,0 +1,74 @@
+/*
+ * Copyright (C) 2014  René Kijewski  <rene.kijewski@fu-berlin.de>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
+ */
+
+/**
+ * @ingroup     pthread
+ * @{
+ *
+ * @file
+ * @brief       Implementation of a fair, POSIX conforming reader/writer lock (attribute set).
+ *
+ * @author      René Kijewski <rene.kijewski@fu-berlin.de>
+ *
+ * @}
+ */
+
+#include "pthread.h"
+
+#include <string.h>
+
+int pthread_rwlockattr_init(pthread_rwlockattr_t *attr)
+{
+    if (attr == NULL) {
+        return EINVAL;
+    }
+
+    memset(attr, 0, sizeof (*attr));
+    return 0;
+}
+
+int pthread_rwlockattr_destroy(pthread_rwlockattr_t *attr)
+{
+    if (attr == NULL) {
+        return EINVAL;
+    }
+
+    (void) attr;
+    return 0;
+}
+
+/* Return current setting of process-shared attribute of ATTR in PSHARED.  */
+int pthread_rwlockattr_getpshared(const pthread_rwlockattr_t *attr, int *pshared)
+{
+    if (attr == NULL || pshared == NULL) {
+        return EINVAL;
+    }
+
+    *pshared = attr->pshared;
+    return 0;
+}
+
+int pthread_rwlockattr_setpshared(pthread_rwlockattr_t *attr, int pshared)
+{
+    if (attr == NULL || (pshared != PTHREAD_PROCESS_SHARED && pshared != PTHREAD_PROCESS_PRIVATE)) {
+        return EINVAL;
+    }
+
+    attr->pshared = pshared;
+    return 0;
+}