2015-08-03 18:52:32 +02:00
|
|
|
/*
|
2016-11-22 18:21:40 +01:00
|
|
|
* Copyright (C) 2016 TriaGnoSys GmbH
|
|
|
|
* 2013-15 Freie Universität Berlin
|
2015-08-03 18:52:32 +02:00
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*
|
2016-11-22 18:21:40 +01:00
|
|
|
* @author Víctor Ariño <victor.arino@zii.aero>
|
2015-08-03 18:52:32 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
#include "irq.h"
|
2016-11-22 18:21:40 +01:00
|
|
|
#include "assert.h"
|
2015-10-19 16:15:37 +02:00
|
|
|
#include "sema.h"
|
2016-11-22 18:21:40 +01:00
|
|
|
#include "xtimer.h"
|
2015-08-03 18:52:32 +02:00
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
2016-11-22 18:21:40 +01:00
|
|
|
void sema_create(sema_t *sema, unsigned int value)
|
2015-08-03 18:52:32 +02:00
|
|
|
{
|
2016-11-22 18:21:40 +01:00
|
|
|
assert(sema != NULL);
|
|
|
|
|
2015-10-19 16:25:06 +02:00
|
|
|
sema->value = value;
|
2016-11-22 18:21:40 +01:00
|
|
|
sema->state = SEMA_OK;
|
|
|
|
mutex_init(&sema->mutex);
|
|
|
|
if (value == 0) {
|
|
|
|
mutex_lock(&sema->mutex);
|
|
|
|
}
|
2015-08-03 18:52:32 +02:00
|
|
|
}
|
|
|
|
|
2016-11-22 18:21:40 +01:00
|
|
|
void sema_destroy(sema_t *sema)
|
2015-08-03 18:52:32 +02:00
|
|
|
{
|
2016-11-22 18:21:40 +01:00
|
|
|
assert(sema != NULL);
|
2015-12-01 19:11:45 +01:00
|
|
|
|
2016-11-22 18:21:40 +01:00
|
|
|
sema->state = SEMA_DESTROY;
|
|
|
|
mutex_unlock(&sema->mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
int sema_try_wait(sema_t *sema)
|
|
|
|
{
|
|
|
|
assert(sema != NULL);
|
|
|
|
|
|
|
|
int ret = -ECANCELED;
|
|
|
|
if (sema->state == SEMA_OK) {
|
|
|
|
ret = -EAGAIN;
|
|
|
|
unsigned old = irq_disable();
|
|
|
|
if (sema->value > 0) {
|
|
|
|
--sema->value;
|
|
|
|
ret = 0;
|
|
|
|
}
|
|
|
|
irq_restore(old);
|
2015-08-03 18:52:32 +02:00
|
|
|
}
|
2016-11-22 18:21:40 +01:00
|
|
|
|
|
|
|
return ret;
|
2015-08-03 18:52:32 +02:00
|
|
|
}
|
|
|
|
|
2016-11-22 18:21:40 +01:00
|
|
|
int sema_wait_timed(sema_t *sema, uint64_t us)
|
2015-08-03 18:52:32 +02:00
|
|
|
{
|
2016-11-22 18:21:40 +01:00
|
|
|
assert(sema != NULL);
|
2015-12-01 19:11:45 +01:00
|
|
|
|
2016-11-22 18:21:40 +01:00
|
|
|
if (sema->state != SEMA_OK) {
|
|
|
|
return -ECANCELED;
|
2015-08-03 18:52:32 +02:00
|
|
|
}
|
|
|
|
|
2016-11-22 18:21:40 +01:00
|
|
|
unsigned old = irq_disable();
|
|
|
|
if (sema->value == 0) {
|
|
|
|
irq_restore(old);
|
|
|
|
int timeout = xtimer_mutex_lock_timeout(&sema->mutex, us);
|
2015-08-03 18:52:32 +02:00
|
|
|
|
2016-11-22 18:21:40 +01:00
|
|
|
if (sema->state != SEMA_OK) {
|
|
|
|
mutex_unlock(&sema->mutex);
|
|
|
|
return -ECANCELED;
|
2015-08-03 18:52:32 +02:00
|
|
|
}
|
|
|
|
|
2016-11-22 18:21:40 +01:00
|
|
|
if (timeout) {
|
|
|
|
return -ETIMEDOUT;
|
2015-08-03 18:52:32 +02:00
|
|
|
}
|
2016-11-22 18:21:40 +01:00
|
|
|
|
|
|
|
old = irq_disable();
|
2015-08-03 18:52:32 +02:00
|
|
|
}
|
|
|
|
|
2016-11-22 18:21:40 +01:00
|
|
|
unsigned int value = --sema->value;
|
|
|
|
irq_restore(old);
|
|
|
|
|
|
|
|
if (value > 0) {
|
|
|
|
mutex_unlock(&sema->mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2015-08-03 18:52:32 +02:00
|
|
|
}
|
|
|
|
|
2015-10-19 16:25:06 +02:00
|
|
|
int sema_post(sema_t *sema)
|
2015-08-03 18:52:32 +02:00
|
|
|
{
|
2016-11-22 18:21:40 +01:00
|
|
|
assert(sema != NULL);
|
2015-12-01 19:11:45 +01:00
|
|
|
|
2016-11-22 18:21:40 +01:00
|
|
|
unsigned old = irq_disable();
|
|
|
|
if (sema->value == UINT_MAX) {
|
|
|
|
irq_restore(old);
|
2015-08-03 18:52:32 +02:00
|
|
|
return -EOVERFLOW;
|
|
|
|
}
|
2016-11-22 18:21:40 +01:00
|
|
|
|
|
|
|
unsigned value = sema->value++;
|
|
|
|
irq_restore(old);
|
|
|
|
|
|
|
|
if (value == 0) {
|
|
|
|
mutex_unlock(&sema->mutex);
|
2015-08-03 18:52:32 +02:00
|
|
|
}
|
|
|
|
|
2015-12-26 17:30:35 +01:00
|
|
|
return 0;
|
2015-08-03 18:52:32 +02:00
|
|
|
}
|
|
|
|
/** @} */
|