2014-04-17 14:20:46 +02:00
|
|
|
/*
|
|
|
|
* 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 tests
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Test rwlock implementation.
|
|
|
|
*
|
|
|
|
* @author René Kijewski <rene.kijewski@fu-berlin.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <pthread.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#include "random.h"
|
|
|
|
#include "sched.h"
|
|
|
|
#include "thread.h"
|
2015-09-03 21:36:13 +02:00
|
|
|
#include "xtimer.h"
|
2014-04-17 14:20:46 +02:00
|
|
|
|
|
|
|
#define NUM_READERS_HIGH 2
|
|
|
|
#define NUM_READERS_LOW 3
|
|
|
|
|
|
|
|
#define NUM_WRITERS_HIGH 1
|
|
|
|
#define NUM_WRITERS_LOW 2
|
|
|
|
|
|
|
|
#define NUM_READERS (NUM_READERS_HIGH + NUM_READERS_LOW)
|
|
|
|
#define NUM_WRITERS (NUM_WRITERS_HIGH + NUM_WRITERS_LOW)
|
|
|
|
#define NUM_CHILDREN (NUM_READERS + NUM_WRITERS)
|
|
|
|
|
|
|
|
#define NUM_ITERATIONS 5
|
|
|
|
|
|
|
|
#define RAND_SEED 0xC0FFEE
|
|
|
|
|
|
|
|
static pthread_rwlock_t rwlock;
|
|
|
|
static volatile unsigned counter;
|
|
|
|
|
2017-10-29 12:22:44 +01:00
|
|
|
static kernel_pid_t main_thread_pid;
|
|
|
|
|
2019-07-15 16:59:52 +02:00
|
|
|
/* The test assumes that 'printf/puts' are non interruptible operations
|
|
|
|
* use a mutex to guarantee it */
|
|
|
|
static mutex_t stdout_mutex = MUTEX_INIT;
|
|
|
|
|
|
|
|
#define PRINTF(FMT, ...) \
|
|
|
|
do { \
|
|
|
|
mutex_lock(&stdout_mutex); \
|
|
|
|
printf("%c%" PRIkernel_pid " (prio=%u): " FMT "\n", \
|
|
|
|
__func__[0], sched_active_pid, \
|
|
|
|
sched_active_thread->priority, \
|
|
|
|
(int)__VA_ARGS__); \
|
|
|
|
mutex_unlock(&stdout_mutex); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define PUTS(s) \
|
|
|
|
do { \
|
|
|
|
mutex_lock(&stdout_mutex); \
|
|
|
|
puts(s); \
|
|
|
|
mutex_unlock(&stdout_mutex); \
|
|
|
|
} while (0)
|
2017-10-29 12:22:44 +01:00
|
|
|
|
|
|
|
static void _notify_main_thread(void)
|
|
|
|
{
|
|
|
|
msg_t msg;
|
|
|
|
msg_send(&msg, main_thread_pid);
|
|
|
|
}
|
2014-04-17 14:20:46 +02:00
|
|
|
|
|
|
|
static void do_sleep(int factor)
|
|
|
|
{
|
2016-02-14 20:04:10 +01:00
|
|
|
uint32_t timeout_us = (random_uint32() % 100000) * factor;
|
2017-10-29 12:22:44 +01:00
|
|
|
PRINTF("sleep for % 8i µs.", timeout_us);
|
2015-09-03 21:36:13 +02:00
|
|
|
xtimer_usleep(timeout_us);
|
2014-04-17 14:20:46 +02:00
|
|
|
}
|
|
|
|
|
2014-03-04 20:20:01 +01:00
|
|
|
static void *writer(void *arg)
|
2014-04-17 14:20:46 +02:00
|
|
|
{
|
2014-03-04 20:20:01 +01:00
|
|
|
(void) arg;
|
2019-07-15 16:59:52 +02:00
|
|
|
PUTS("start");
|
2017-10-29 12:22:44 +01:00
|
|
|
for (unsigned i = 0; i < NUM_ITERATIONS; ++i) {
|
2014-04-17 14:20:46 +02:00
|
|
|
pthread_rwlock_wrlock(&rwlock);
|
|
|
|
unsigned cur = ++counter;
|
|
|
|
do_sleep(3); /* simulate time that it takes to write the value */
|
|
|
|
PRINTF("%i: write -> %2u (correct = %u)", i, cur, cur == counter);
|
|
|
|
pthread_rwlock_unlock(&rwlock);
|
|
|
|
do_sleep(2);
|
|
|
|
}
|
2019-07-15 16:59:52 +02:00
|
|
|
PUTS("done");
|
2017-10-29 12:22:44 +01:00
|
|
|
_notify_main_thread();
|
2014-03-04 20:20:01 +01:00
|
|
|
return NULL;
|
2014-04-17 14:20:46 +02:00
|
|
|
}
|
|
|
|
|
2014-03-04 20:20:01 +01:00
|
|
|
static void *reader(void *arg)
|
2014-04-17 14:20:46 +02:00
|
|
|
{
|
2014-03-04 20:20:01 +01:00
|
|
|
(void) arg;
|
2019-07-15 16:59:52 +02:00
|
|
|
PUTS("start");
|
2017-10-29 12:22:44 +01:00
|
|
|
for (unsigned i = 0; i < NUM_ITERATIONS; ++i) {
|
2014-04-17 14:20:46 +02:00
|
|
|
pthread_rwlock_rdlock(&rwlock);
|
|
|
|
unsigned cur = counter;
|
|
|
|
do_sleep(1); /* simulate time that it takes to read the value */
|
|
|
|
PRINTF("%i: read <- %2u (correct = %u)", i, cur, cur == counter);
|
|
|
|
pthread_rwlock_unlock(&rwlock);
|
|
|
|
do_sleep(1);
|
|
|
|
}
|
2019-07-15 16:59:52 +02:00
|
|
|
PUTS("done");
|
2017-10-29 12:22:44 +01:00
|
|
|
_notify_main_thread();
|
|
|
|
|
2014-03-04 20:20:01 +01:00
|
|
|
return NULL;
|
2014-04-17 14:20:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int main(void)
|
|
|
|
{
|
2015-04-28 20:02:05 +02:00
|
|
|
static char stacks[NUM_CHILDREN][THREAD_STACKSIZE_MAIN];
|
2014-04-17 14:20:46 +02:00
|
|
|
|
2019-07-15 16:59:52 +02:00
|
|
|
PUTS("START");
|
2017-10-29 12:22:44 +01:00
|
|
|
/* Get main thread pid */
|
|
|
|
main_thread_pid = thread_getpid();
|
2014-04-17 14:20:46 +02:00
|
|
|
|
|
|
|
for (unsigned i = 0; i < NUM_CHILDREN; ++i) {
|
|
|
|
int prio;
|
2014-03-04 20:20:01 +01:00
|
|
|
void *(*fun)(void *);
|
2014-04-17 14:20:46 +02:00
|
|
|
const char *name;
|
|
|
|
|
|
|
|
if (i < NUM_READERS) {
|
|
|
|
if (i < NUM_READERS_HIGH) {
|
2015-04-28 20:02:05 +02:00
|
|
|
prio = THREAD_PRIORITY_MAIN + 1;
|
2014-04-17 14:20:46 +02:00
|
|
|
}
|
|
|
|
else {
|
2015-04-28 20:02:05 +02:00
|
|
|
prio = THREAD_PRIORITY_MAIN + 2;
|
2014-04-17 14:20:46 +02:00
|
|
|
}
|
|
|
|
fun = reader;
|
|
|
|
name = "reader";
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (i - NUM_READERS < NUM_WRITERS_HIGH) {
|
2015-04-28 20:02:05 +02:00
|
|
|
prio = THREAD_PRIORITY_MAIN + 1;
|
2014-04-17 14:20:46 +02:00
|
|
|
}
|
|
|
|
else {
|
2015-04-28 20:02:05 +02:00
|
|
|
prio = THREAD_PRIORITY_MAIN + 2;
|
2014-04-17 14:20:46 +02:00
|
|
|
}
|
|
|
|
fun = writer;
|
|
|
|
name = "writer";
|
|
|
|
}
|
|
|
|
|
2014-03-04 20:20:01 +01:00
|
|
|
thread_create(stacks[i], sizeof(stacks[i]),
|
2015-12-02 12:00:37 +01:00
|
|
|
prio, THREAD_CREATE_WOUT_YIELD | THREAD_CREATE_STACKTEST,
|
2014-03-04 20:20:01 +01:00
|
|
|
fun, NULL, name);
|
2014-04-17 14:20:46 +02:00
|
|
|
}
|
|
|
|
|
2017-10-29 12:22:44 +01:00
|
|
|
/* Block until all children threads are done */
|
|
|
|
for (unsigned i = 0; i < NUM_CHILDREN; ++i) {
|
|
|
|
msg_t msg;
|
|
|
|
msg_receive(&msg);
|
|
|
|
}
|
|
|
|
|
2019-07-15 16:59:52 +02:00
|
|
|
PUTS("SUCCESS");
|
2017-10-29 12:22:44 +01:00
|
|
|
|
2014-04-17 14:20:46 +02:00
|
|
|
return 0;
|
|
|
|
}
|