2013-11-27 16:28:31 +01:00
|
|
|
/*
|
2014-04-07 11:16:32 +02:00
|
|
|
* Copyright (C) 2014 Freie Universität Berlin
|
2015-04-28 22:57:56 +02:00
|
|
|
* Copyright (C) 2015 Eistec AB
|
2010-09-22 15:10:42 +02:00
|
|
|
*
|
2014-08-23 15:43:13 +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.
|
2013-11-27 16:28:31 +01:00
|
|
|
*/
|
2014-02-11 18:15:43 +01:00
|
|
|
|
2013-11-27 16:28:31 +01:00
|
|
|
/**
|
|
|
|
* @addtogroup core_util
|
2010-09-22 15:10:42 +02:00
|
|
|
* @{
|
2013-11-27 16:28:31 +01:00
|
|
|
*
|
2015-04-28 22:57:56 +02:00
|
|
|
* @file
|
|
|
|
* @brief Functions for atomic handling of variables
|
2013-11-27 16:28:31 +01:00
|
|
|
*
|
2014-01-28 11:50:12 +01:00
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
2015-09-20 13:47:39 +02:00
|
|
|
* @author Joakim Nohlgård <joakim.nohlgard@eistec.se>
|
2010-09-22 15:10:42 +02:00
|
|
|
*/
|
|
|
|
|
2017-01-18 13:00:05 +01:00
|
|
|
#ifndef ATOMIC_H
|
|
|
|
#define ATOMIC_H
|
2010-09-22 15:10:42 +02:00
|
|
|
|
2014-10-09 01:18:16 +02:00
|
|
|
#ifdef __cplusplus
|
2015-04-28 22:57:56 +02:00
|
|
|
extern "C" {
|
2014-10-09 01:18:16 +02:00
|
|
|
#endif
|
|
|
|
|
2015-04-28 22:57:56 +02:00
|
|
|
/**
|
|
|
|
* @brief Integer variable for use in atomic counters.
|
2016-02-16 13:43:10 +01:00
|
|
|
*
|
|
|
|
* @note This type is a struct for hard type checking (let the compiler warn
|
|
|
|
* if int is assigned regularly).
|
2015-04-28 22:57:56 +02:00
|
|
|
*/
|
2016-04-07 21:19:04 +02:00
|
|
|
typedef struct {
|
2015-12-02 11:14:07 +01:00
|
|
|
volatile int value; /**< the actual value */
|
2015-04-28 22:57:56 +02:00
|
|
|
} atomic_int_t;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Initializer for atomic variables
|
|
|
|
*
|
|
|
|
* @param[in] val initial value for the atomic integer
|
|
|
|
*/
|
|
|
|
#define ATOMIC_INIT(val) {(val)}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Atomic Compare and Swap
|
|
|
|
*
|
|
|
|
* Updates the value in var iff the current value of var is equal to old.
|
|
|
|
*
|
|
|
|
* @param[inout] var Atomic variable to update
|
|
|
|
* @param[in] old The old value to compare against
|
|
|
|
* @param[in] now The new value to write to var
|
|
|
|
*
|
|
|
|
* @return 1 if the write completed successfully
|
|
|
|
* @return 0 if the write failed.
|
|
|
|
*/
|
|
|
|
int atomic_cas(atomic_int_t *var, int old, int now);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Increment a counter variable by one atomically and return the old value.
|
|
|
|
*
|
|
|
|
* @param[inout] var Pointer to a counter variable.
|
|
|
|
*
|
|
|
|
* @return The value of *val* before the increment.
|
|
|
|
*/
|
|
|
|
static inline int atomic_inc(atomic_int_t *var)
|
|
|
|
{
|
|
|
|
int old;
|
|
|
|
|
|
|
|
do {
|
|
|
|
old = var->value;
|
|
|
|
} while (!atomic_cas(var, old, old + 1));
|
|
|
|
|
|
|
|
return old;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Decrement a counter variable by one atomically and return the old value.
|
|
|
|
*
|
|
|
|
* @param[inout] var Pointer to a counter variable.
|
|
|
|
*
|
|
|
|
* @return The value of *val* before the decrement.
|
|
|
|
*/
|
|
|
|
static inline int atomic_dec(atomic_int_t *var)
|
|
|
|
{
|
|
|
|
int old;
|
|
|
|
|
|
|
|
do {
|
|
|
|
old = var->value;
|
|
|
|
} while (!atomic_cas(var, old, old - 1));
|
|
|
|
|
|
|
|
return old;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Set an atomic variable to 1.
|
|
|
|
*
|
|
|
|
* @param[inout] var Pointer to an atomic variable to update
|
|
|
|
*
|
|
|
|
* @return 1 if the old value was 0 and the variable was successfully updated
|
|
|
|
* @return 0 if the variable was already set
|
|
|
|
*/
|
|
|
|
static inline int atomic_set_to_one(atomic_int_t *var)
|
|
|
|
{
|
|
|
|
do {
|
|
|
|
if (var->value != 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
} while (!atomic_cas(var, 0, 1));
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Set an atomic variable to 0.
|
|
|
|
*
|
|
|
|
* @param[inout] var Pointer to an atomic variable to update
|
|
|
|
*
|
|
|
|
* @return 1 if the old value was not 0 and the variable was successfully updated
|
|
|
|
* @return 0 if the variable was already cleared
|
|
|
|
*/
|
|
|
|
static inline int atomic_set_to_zero(atomic_int_t *var)
|
|
|
|
{
|
|
|
|
int old;
|
|
|
|
|
|
|
|
do {
|
|
|
|
old = var->value;
|
|
|
|
|
|
|
|
if (old == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
} while (!atomic_cas(var, old, 0));
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Get the value of an atomic int
|
|
|
|
*
|
|
|
|
* @param[in] var Atomic variable
|
|
|
|
*
|
|
|
|
* @return the value of the atomic integer
|
|
|
|
*
|
|
|
|
* @note This can be used for non-thread-safe assignment of the atomic integer
|
|
|
|
*/
|
|
|
|
#define ATOMIC_VALUE(var) ((var).value)
|
|
|
|
|
2014-10-09 01:18:16 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-01-18 13:00:05 +01:00
|
|
|
#endif /* ATOMIC_H */
|
2014-04-07 11:16:32 +02:00
|
|
|
/** @} */
|